Spring Hibernate:重载实体映射

时间:2014-03-11 09:10:25

标签: java spring hibernate spring-mvc

在Web应用程序上,我们使用Spring 3.2和Hibernate 4.1.1并实现类似插件的体系结构。 可以在运行时添加和删除插件。 对于每个模块,我们定义了一个单独的类加载器,并在spring上创建单独的子应用程 完成配置是使用注释完成的,不再为bean提供XML配置。

Spring Hibernate配置类

@Configuration
@EnableTransactionManagement
public class HibernateConfigurationFactory {

@Bean
public JndiObjectFactoryBean dataSource() {
    JndiObjectFactoryBean ds = new JndiObjectFactoryBean();
    ds.setJndiName("java:jboss/datasources/OurOwnDS");
    ds.setResourceRef(true);
    return ds;
}

@Bean
public LocalSessionFactoryBean sessionFactory() {
    LocalSessionFactoryBean sessionFactory = new LocalSessionFactoryBean();
    sessionFactory.setPackagesToScan("com.foo.bar");
    sessionFactory.setDataSource((DataSource) dataSource().getObject());

    Properties hibernateProperties = new Properties();
    hibernateProperties.put("hibernate.hbm2ddl.auto", "update");

    sessionFactory.setHibernateProperties(hibernateProperties);

    return sessionFactory;
}

@Bean
public HibernateTransactionManager transactionManager() {
    HibernateTransactionManager transactionManager = new HibernateTransactionManager();
    transactionManager.setSessionFactory(sessionFactory().getObject());
    return transactionManager;
}

}

现在问题: 一些插件包含自己的实体(+ DAO)类,这些类在运行时与模块一起添加。

是否可以在hibernate上创建某种单独的上下文(就像我们在spring上做的那样),甚至可以添加/重新加载其他实体类?

重新加载EntityManager是否符合我的需求? 但是在上下文中已经加载的实体会发生什么?

感谢您提前提供任何帮助和评论。

更新 实际上我做了以下事情并解决了这个问题(但后来遇到了另一个问题)。

我为每个模块/上下文创建一个新的DataSource + SessionFactory + TransactionManager,并将它们插入到新的子ApplicationContext中。 现在我使用类加载器扫描所有带注释的类,并使用

在应用程序上下文和会话工厂中手动注册它们
LocalSessionFactorybean#setAnnotatedClasses(...)

这很有效......但是......

下一个问题: 我得到一个ClassNotFoundException似乎是一个类加载器问题。 Hibernate使用系统类加载器而不是我自己的pluginClassloader。

有人知道如何将自己的类加载器注入Hibernate吗?

2 个答案:

答案 0 :(得分:2)

注入Environment并将其用作Hibernate资源的额外源是否可行?

@Configuration
@EnableTransactionManagement
public class HibernateConfigurationFactory {

    @Autowired
    Environment env;

    @Bean 
    public LocalSessionFactoryBean sessionFactory() {
        LocalSessionFactoryBean sessionFactory = new LocalSessionFactoryBean();
        sessionFactory.setPackagesToScan("com.foo.bar");
        if (env.containsProperty("some.extra.classes.property") {
            sessionFactory.setAnnotatedClasses(some extrapolation here);
            // Or similarly add extra packages for scanning
            ...
        }
    }
}

对于额外的DAO bean,您可以使用@Profile,或者 - 如果使用Spring 4 - 使用@Conditional

修改

Environment不是你创造的东西,它是"那里有你" 。它基本上是Property Source和Profile集的容器。

一个很好的参考点是Spring参考文档,即IoC容器章节。您还可以查看SpringSource博客。 Chris Beams很少有关于Spring 3.1的好文章,但大部分内容都在那里。

例如,您可以使用以下内容来引导子应用程序上下文:

AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
ConfigurableEnvironment environment = context.getEnvironment();

// this is how you set different properties per a sub context. 
Map subContextMap = new HashMap();
subContextMap.put("some.extra.classes.property", [unique value here]);
environment.getPropertySources().addFirst(new MapPropertySource("SUB_CTX_MAP", subContextMap);

// this is the generic configuration class(es).
context.register(HibernateConfigurationFactory.class);

context.refresh();

答案 1 :(得分:1)

如果有人需要类似的解决方案,只需更新。

我通过以下方式修复了这种情况:

  • 使用当前根上下文创建新的ApplicationContext 父
  • 将新的类加载器引入上下文和自己的Resource-Loader
  • 以编程方式我为新上下文创建一个新的会话工厂和事务管理器

        // create the datasource bean
        BeanDefinitionBuilder dataSourceBeanBuilder = BeanDefinitionBuilder.rootBeanDefinition(DataSourceConfiguration.class, "createDataSource");
        dataSourceBeanBuilder.addConstructorArgValue(descriptor.getDataSourceDescriptor().getJNDILookupName());
        dataSourceBeanBuilder.addConstructorArgValue(descriptor.getDataSourceDescriptor().isResourceRef());
        moduleContext.registerBeanDefinition("dataSource", dataSourceBeanBuilder.getBeanDefinition());
    
        // now build the sessionFactor
        BeanDefinitionBuilder sessionFactoryBeanBuilder = BeanDefinitionBuilder.rootBeanDefinition(SessionFactoryFactory.class, "createSessionFactory");
        sessionFactoryBeanBuilder.addConstructorArgReference("dataSource");
        sessionFactoryBeanBuilder.addConstructorArgValue(module.getKey());
        sessionFactoryBeanBuilder.addConstructorArgValue(moduleContext.getModuleResourceLoader());
        sessionFactoryBeanBuilder.addConstructorArgValue(annotatedClasses);
        moduleContext.registerBeanDefinition("sessionFactory", sessionFactoryBeanBuilder.getBeanDefinition());
    
        // now build the transactionManager
        BeanDefinitionBuilder transactionManagerBeanBuilder = BeanDefinitionBuilder.rootBeanDefinition(HibernateConfigurationFactory.class, "createTransactionManager");
        transactionManagerBeanBuilder.addConstructorArgReference("sessionFactory");
        moduleContext.registerBeanDefinition("transactionManager", transactionManagerBeanBuilder.getBeanDefinition());
    
    • 调用context.refresh()来加载和初始化所有实体
    • 调用context.start()以启动新上下文