在JPA / Hibernate中使用Spring定义的transactionManager

时间:2010-09-29 08:43:32

标签: java hibernate spring jpa jta

假设您将JPA与Spring一起使用,并将Hibernate用作JPA实现。 JPA事务模式是“JTA”,因此您需要将容器transactionManager传递给Hibernate。 经典的答案是将hibernate.transaction.manager_lookup_class设置为服务器的匹配类。

但是,我认为让这取决于特定于服务器的配置是一种耻辱,因为您已经在Spring中找到了<tx:jta-transaction-manager>的事务管理器。

有没有办法通过类似

的配置将此事务管理器提供给Hibernate
 <bean id="entityManagerFactory"
  class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
  <property name="persistenceUnitName" value="persistence_unit_name"/>
  <property name="jpaVendorAdapter">
   <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"/>
  </property>
  <property name="jpaProperties">
   <props>
    <prop key="hibernate.transaction.manager_lookup_class">
     org.hibernate.transaction.SunONETransactionManagerLookup
    </prop>
   </props>
  </property>
 </bean>

 <tx:jta-transaction-manager/>

目标是摆脱org.hibernate.transaction.SunONETransactionManagerLookup财产。 顺便说一句,我真的有两种不同的服务器实现。

编辑:没有事务管理器配置,Hibernate在创建EntityManagerFactory时会窒息:

Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'entityManagerFactory' defined in URL [file:/C:/configuration/afoCuad-metier-ear/entitymanager-base-context.xml]: Invocation of init method failed; nested exception is javax.persistence.PersistenceException: [PersistenceUnit: fr.tm.ima.cuad-afoCuad-metier-ejb-PU] Unable to build EntityManagerFactory
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1420)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:519)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:456)
at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:291)
at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:222)
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:288)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:190)
at org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor.findDefaultEntityManagerFactory(PersistenceAnnotationBeanPostProcessor.java:529)
at org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor.findEntityManagerFactory(PersistenceAnnotationBeanPostProcessor.java:495)
at org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor$PersistenceElement.resolveEntityManager(PersistenceAnnotationBeanPostProcessor.java:656)
at org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor$PersistenceElement.getResourceToInject(PersistenceAnnotationBeanPostProcessor.java:629)
at org.springframework.beans.factory.annotation.InjectionMetadata$InjectedElement.inject(InjectionMetadata.java:147)
at org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:84)
at org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor.postProcessPropertyValues(PersistenceAnnotationBeanPostProcessor.java:338)
... 80 more
Caused by: javax.persistence.PersistenceException: [PersistenceUnit: fr.tm.ima.cuad-afoCuad-metier-ejb-PU] Unable to build EntityManagerFactory
at org.hibernate.ejb.Ejb3Configuration.buildEntityManagerFactory(Ejb3Configuration.java:901)
at org.hibernate.ejb.HibernatePersistence.createContainerEntityManagerFactory(HibernatePersistence.java:74)
at org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean.createNativeEntityManagerFactory(LocalContainerEntityManagerFactoryBean.java:225)
at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.afterPropertiesSet(AbstractEntityManagerFactoryBean.java:308)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1477)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1417)
... 93 more
Caused by: org.hibernate.HibernateException: The chosen transaction strategy requires access to the JTA TransactionManager
at org.hibernate.impl.SessionFactoryImpl.<init>(SessionFactoryImpl.java:401)
at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:1385)
at org.hibernate.cfg.AnnotationConfiguration.buildSessionFactory(AnnotationConfiguration.java:954)
at org.hibernate.ejb.Ejb3Configuration.buildEntityManagerFactory(Ejb3Configuration.java:892)
... 98 more

3 个答案:

答案 0 :(得分:5)

首先 - 你真的需要JTA吗?通常spring + hibernate不需要它。您可以使用简单的JpaTransactionManager / HibernateTransactionManager

如果您真的想要JTA,那么您将需要一个JTA提供商。如果未在应用程序服务器中运行,请检查this question以了解如何在servlet容器中使用JTA。 (另请查看this question

最后,hibernate docs指定对于容器管理的事务:

  

声明式事务划分是EJB的标准功能,也称为容器管理事务(CMT)。在EJB 2.x中,您将使用XML部署描述符来创建事务组合。在EJB 3.x中,您可以直接在源代码中使用JDK 5.0注释元数据,这是一种不那么冗长的方法。在Hibernate配置中为EJB启用CMT事务划分:

     
      
  • hibernate.transaction.manager_lookup_class设置为JEE容器的查找策略
  •   
  • hibernate.transaction.factory_class设为org.hibernate.transaction.CMTTransactionFactory
  •   

第二点可能是你错过了什么?

除了那个文档(下一节)之外,文档说的是如果你想要声明式事务,那么hibernate就不是你应该看的地方了。你需要创建一个拦截器。这正是春季交易经理所追求的。鉴于您的技术堆栈(Spring),这将是我的选择。

如果您不想依赖单个JTA提供程序,请进行两次构建。例如,maven有“maven profiles”,允许为不同的环境进行构建。

答案 1 :(得分:1)

不幸的是,如果像许多其他JBoss产品那样查看Hibernate API,他们就会有一个通常称为Configuration的类来保存大部分(如果不是全部)主配置。不幸的是,他们(JBoss)似乎喜欢为参数和类保存“Strings”来定位实例。几乎总是不可能简单地设置一个实际的预制作准备设置。

我将尝试尝试类似于以下的内容,原因与您提到的相同。

  • 创建TransactionManagerLookup的实现
  • 包含一个setter,它接受一个TM并设置一个线程局部变量+ instance。
  • 在传递给配置的属性中传递TML的名称。
  • 当您的TML启动时,将线程局部变量复制到您的实例fie.d。
  • 一旦一切都完成,
  • 清除threadlocal。

答案 2 :(得分:0)

我最近用JPA / Grails做了一些事情,我使用的配置是这样的:

这有帮助吗?

<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
    <property name="dataSource" ref="dataSource"/>
    <property name="entityManagerFactory" ref="entityManagerFactory"/>
</bean>

<bean id="entityManagerFactory" class="org.hibernate.ejb.EntityManagerFactoryImpl">
 <constructor-arg index="0" ref="sessionFactory"/>
 <constructor-arg index="1">
   <bean id="javax.persistence.spi.PersistenceUnitTransactionType.RESOURCE_LOCAL" class="org.springframework.beans.factory.config.FieldRetrievingFactoryBean" />
 </constructor-arg>
 <constructor-arg index="2" value="true"/>
 <constructor-arg index="3"><null/></constructor-arg>
</bean>