org.hibernate.HibernateException:没有在休眠升级时指定 TransactionManagerLookup
Posted
技术标签:
【中文标题】org.hibernate.HibernateException:没有在休眠升级时指定 TransactionManagerLookup【英文标题】:org.hibernate.HibernateException: No TransactionManagerLookup specified on hibernate upgrade 【发布时间】:2015-03-21 05:02:57 【问题描述】:我正在运行 spring 4.1.4、hibernate 4.3.8、atomikos 3.9.3、java 8、tomcat 8。
当我启动我的服务器时,我在 localhost.log 中看到了上述异常,但是除了我已经配置它的位置之外,我不确定在哪里配置 TransactionManagerLookup。这在升级 hibernate 之前没有发生,因此很可能是版本控制问题。有人可以帮忙吗?
仅供参考:catalina.out 没有任何用处。只是:
SEVERE [localhost-startStop-1] org.apache.catalina.core.StandardContext.startInternal Error listenerStart
SEVERE [localhost-startStop-1] org.apache.catalina.core.StandardContext.startInternal Context [] startup failed due to previous errors
我的 appContext 定义:
<prop key="hibernate.transaction.factory_class">org.hibernate.engine.transaction.internal.jta.CMTTransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
完整的堆栈跟踪是:
22-Jan-2015 10:07:25.734 SEVERE [localhost-startStop-1] org.apache.catalina.core.StandardContext.listenerStart Exception sending context initialized event to listener instance of class com.my.app.web.InitializerListener
org.hibernate.HibernateException: No TransactionManagerLookup specified
at org.hibernate.context.internal.JTASessionContext.currentSession(JTASessionContext.java:85)
at org.hibernate.internal.SessionFactoryImpl.getCurrentSession(SessionFactoryImpl.java:1014)
at com.my.app.dao.AbstractMyDAO.currentSession(AbstractMyDAO.java:116)
at com.my.app.dao.AbstractMyDAO.criteria(AbstractMyDAO.java:86)
at com.my.app.dao.AbstractMyDAO.count(AbstractMyDAO.java:79)
at com.my.app.initialize.MyInitializerImpl.initializeApplicaiton(MyInitializerImpl.java:69)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:483)
at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:317)
at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:190)
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:157)
at org.springframework.transaction.interceptor.TransactionInterceptor$1.proceedWithInvocation(TransactionInterceptor.java:99)
at org.springframework.transaction.interceptor.TransactionAspectSupport.invokeWithinTransaction(TransactionAspectSupport.java:281)
at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:96)
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)
at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:207)
at com.sun.proxy.$Proxy220.initializeApplicaiton(Unknown Source)
at com.my.app.web.MyInitializerListener.contextInitialized(MyInitializerListener.java:42)
at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:4772)
at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5196)
at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
at org.apache.catalina.core.ContainerBase.addChildInternal(ContainerBase.java:725)
at org.apache.catalina.core.ContainerBase.addChild(ContainerBase.java:701)
at org.apache.catalina.core.StandardHost.addChild(StandardHost.java:714)
at org.apache.catalina.startup.HostConfig.deployWAR(HostConfig.java:917)
at org.apache.catalina.startup.HostConfig$DeployWar.run(HostConfig.java:1701)
at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
at java.lang.Thread.run(Thread.java:745)
编辑:Atomikos 事务管理器是这样配置的
<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close">
<property name="forceShutdown" value="true" />
<!-- in secs -->
<property name="transactionTimeout" value="300"/>
</bean>
<bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
<!-- in secs -->
<property name="transactionTimeout" value="300" />
</bean>
<bean id="transactionManager" class="org.springframework.transaction.jta.JtaTransactionManager">
<property name="transactionManager" ref="atomikosTransactionManager" />
<property name="userTransaction" ref="atomikosUserTransaction" />
<property name="transactionSynchronizationName" value="SYNCHRONIZATION_ON_ACTUAL_TRANSACTION" />
<property name="allowCustomIsolationLevels" value="true"/>
</bean>
编辑 2:
我想我需要在这里澄清一下。 我有一个名为 AbstractMyDAO 的 DAO 对象(正如您从堆栈中看到的那样)。在这个对象中定义了一个会话工厂
@Autowired
private SessionFactory mySessionFactory;
当调用 currentSession 时,会抛出上述错误,因为 hibernate 找不到与此会话工厂关联的事务管理器查找。
public Session currentSession()
return mySessionFactory.getCurrentSession();
这个会话工厂在我的应用上下文文件中定义如下:
<bean id="mySessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<property name="dataSource">
<ref bean="myDataSource" />
</property>
<property name="annotatedClasses">
<list>
<value>some values... </value>
...
</list>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">
$gst.hibernate.dialect
</prop>
<prop key="query.factory_class">
org.hibernate.hql.ast.ASTQueryTranslatorFactory
</prop>
<prop key="hibernate.max_fetch_depth">3</prop>
<prop key="hibernate.show_sql">$my.hibernate.showsql</prop>
<prop key="hibernate.cache.use_query_cache">true</prop>
<prop key="hibernate.cache.use_second_level_cache">true</prop>
<prop key="hibernate.cache.use_structured_entries">true</prop>
<prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory</prop>
<prop key="net.sf.ehcache.configurationResourceName">$ehcache.my.persist.config</prop>
<prop key="hibernate.connection.isolation">3</prop>
<prop key="connection.release_mode">auto</prop>
<prop key="hibernate.current_session_context_class">jta</prop>
<prop key="hibernate.transaction.factory_class">org.hibernate.engine.transaction.internal.jta.CMTTransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
<prop key="hibernate.search.default.directory_provider">filesystem</prop>
<prop key="hibernate.search.default.indexBase">/var/log/my/lucene/indexes</prop>
</props>
</property>
</bean>
如您所见,正在定义属性 hibernate.transaction.manager_lookup_class 但在创建 bean 时找不到,我不知道为什么。配置有变化吗?
编辑 3: 当我调试时: final JtaPlatform jtaPlatform = factory().getServiceRegistry().getService(JtaPlatform.class);
它返回一个 NoJTAPlatform。我想这是我的问题。
编辑 4: 似乎没有任何东西可以实现适用于 tomcat 的 JtaPlatform 或者我是不是很精神?
【问题讨论】:
【参考方案1】:我解决了
进入hibernate.cfg.xml
文件
改变
<property name="hibernate.current_session_context_class">thread</property>
到 thread
而不是 jta
。
更多详情here
【讨论】:
【参考方案2】:我的问题的答案是here。
“在 Hibernate 4.3 中,长期弃用的 TransactionManagerLookup 已被删除。现在 JTA 提供程序必须实现 org.hibernate.engine.transaction.jta.platform.spi.JtaPlatform。”
【讨论】:
【参考方案3】:确保您还配置了 Atomikos 事务管理器,以便 TransactionManagerLookup
可以相应地找到 UserTransaction
:
<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager"
init-method="init" destroy-method="close">
<property name="transactionTimeout" value="300" />
</bean>
<bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
<property name="transactionTimeout" value="300" />
</bean>
<bean id="transactionManager"
class="org.springframework.transaction.jta.JtaTransactionManager">
<property name="transactionManager" ref="atomikosTransactionManager" />
<property name="userTransaction" ref="atomikosUserTransaction" />
</bean>
那么,你也应该使用 Spring EntityManager 工厂:
<bean id="entityManager"
class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"
p:dataSource-ref="dataSource"
p:persistenceXmlLocation="**/persistence.xml"
p:persistenceUnitName="persistenceUnit"
depends-on="dataSource, transactionManager">
<property name="jpaVendorAdapter">
<bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
</bean>
</property>
<property name="jpaProperties">
<props>
<prop key="hibernate.transaction.factory_class">org.hibernate.engine.transaction.internal.jta.CMTTransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
...
</props>
</property>
</bean>
那么 DAO 应该注入 EntityManager:
@PersistenceContext(unitName = "persistenceUnit")
private EntityManager entityManager;
而不是直接调用 currentSession(AbstractMyDAO):
at com.my.app.dao.AbstractMyDAO.currentSession(AbstractMyDAO.java:116)
【讨论】:
谢谢,我已经配置了 atomikos 事务管理器(见编辑),但没有配置实体管理器。我会试一试并报告:) 你能解释一下为什么我需要使用实体管理器吗?我的应用程序以前在没有此组件的情况下工作。该问题仅在休眠升级后发生。谢谢。 那是因为 Hibernate 已经朝着成为 JPA 实现的方向发展,而不是被用作独立的 ORM 框架。您仍然可以使用 SessionFactory,但最好使用 SpringLocalSessionFactoryBean
,因为您也可以获得自动 Session 和 JDBC 连接管理。
所以,假设我仍想使用 SessionFactory 而不是 LocalSessionFactoryBean。你知道我目前做错了什么吗?
我没有别的想法。以上是关于org.hibernate.HibernateException:没有在休眠升级时指定 TransactionManagerLookup的主要内容,如果未能解决你的问题,请参考以下文章