因此,在完成一个大型重构项目之后,我就遇到了这个例外,并且不确定如何纠正它。它正在处理一些我未编写的代码,我不熟悉它们的工作方式。还有其他有关此异常的问题,但似乎没有一个适合我的情况。

使用EntityManager的类是SpecialClaimsCaseRepositoryImpl:

package com.redacted.sch.repository.jpa;

//Imports

@Repository
public class SpecialClaimsCaseRepositoryImpl extends SimpleJpaRepository<SpecialClaimsCaseDto, SpecialClaimsCaseDto.Id> implements SpecialClaimsCaseRepository{

    @PersistenceContext(unitName = "schManager")
    private EntityManager em;

          //Some autogenerated methods

    public void setEntityManager(EntityManager em) {
        this.em = em;
    }

    public EntityManager getEntityManager() {
        return em;
    }
}

Persistence.xml:
<persistence xmlns="http://java.sun.com/xml/ns/persistence"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_1_0.xsd"
    version="1.0">

    <persistence-unit name="schManager">
        <provider>org.hibernate.ejb.HibernatePersistence</provider>
        <jta-data-source>jdbc/SCH_DS</jta-data-source>
        <class>com.redacted.sch.domain.model.SpecialClaimsCaseDto</class>
        <properties>
            <property name="hibernate.transaction.manager_lookup_class" value="org.hibernate.transaction.WebSphereExtendedJTATransactionLookup" />
            <property name="hibernate.cache.region.factory_class" value="net.sf.ehcache.hibernate.SingletonEhCacheRegionFactory" />
            <property name="hibernate.cache.use_query_cache" value="true" />
            <property name="hibernate.cache.use_second_level_cache" value="true" />
            <property name="hibernate.dialect" value="com.bcbsks.hibernate.dialect.DB2Dialect" />
            <property name="hibernate.format_sql" value="true" />
            <property name="hibernate.generate_statistics" value="false" />
            <property name="hibernate.jdbc.use_scrollable_resultset" value="true" />
        </properties>
    </persistence-unit>
</persistence>

sch_model_spring.xml:
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">

        <context:component-scan base-package="com.redacted.repository.jpa,
              com.redacted.sch.domain.model,
              com.redacted.sch.repository.jpa,
              com.redacted.sch.service,
              com.redacted.sch.service.impl"/>

        <tx:annotation-driven />

        <tx:jta-transaction-manager />

        <!-- Data source used for testing -->
        <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
            <property name="driverClassName" value="com.ibm.db2.jcc.DB2Driver" />
            <property name="url" value="jdbc:db2:redacted.redacted.com" />
            <property name="username" value="redacted" />
            <property name="password" value="redacted" />
        </bean>

        <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
              <property name="persistenceUnitName" value="schManager" />
              <property name="dataSource" ref="dataSource" />
        </bean>

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

这是我的项目结构:

>

这是堆栈跟踪的一部分,完整跟踪位于此fpaste
Caused by: java.lang.IllegalStateException: A JTA EntityManager cannot use getTransaction()
    at org.hibernate.ejb.AbstractEntityManagerImpl.getTransaction(AbstractEntityManagerImpl.java:985)
    at org.springframework.orm.jpa.DefaultJpaDialect.beginTransaction(DefaultJpaDialect.java:67)
    at org.springframework.orm.jpa.JpaTransactionManager.doBegin(JpaTransactionManager.java:380)
    ... 80 more

我这里真是个菜鸟,所以如果需要其他信息,请询问,我会更新。

感谢您的所有帮助!

最佳答案

问题是您的配置。您已为JTA配置了休眠模式。

<property name="hibernate.transaction.manager_lookup_class" value="org.hibernate.transaction.WebSphereExtendedJTATransactionLookup" />

而您正在使用本地事务而不是分布式事务。
at org.springframework.orm.jpa.JpaTransactionManager.doBegin(JpaTransactionManager.java:380)

您有2种可能的解决方案
  • 删除JpaTransactionManager并将其替换为JTA事务管理器
  • remove从休眠设置中删除hibernate.transaction.manager_lookup_class

  • 如果您真的不需要分布式事务,那么选项2是最简单的,如果您需要分布式事务,只需添加<tx:jta-transaction-manager />即可为您的环境设置适当的JTA tx管理器。删除JpaTransactionManager的定义。

    更新:

    您的配置有两种缺陷。
  • 您的EntityManager配置已经包含用于数据源的jndi查找,您可以通过配置本地数据源
  • 在applicationContext中覆盖该查找
  • 您同时拥有<tx:jta-transaction-manager />JpaTransactionManager,您想使用哪一个?目前,后者正在覆盖第一个。

  • 使用JTA en JNDI查找创建2个单独的配置,一个用于本地测试,一个用于生产。 (最好您的测试代码仅覆盖必需的bean)。

    关于java - Spring IllegalStateException:JTA EntityManager无法使用getTransaction(),我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/24704870/

    10-09 06:07