在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吗?
答案 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)
如果有人需要类似的解决方案,只需更新。
我通过以下方式修复了这种情况:
以编程方式我为新上下文创建一个新的会话工厂和事务管理器
// 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());