为什么我的交易不活跃? javax.persistence.TransactionRequiredException:执行更新/删除查询

时间:2012-06-13 13:48:37

标签: java spring hibernate jboss hibernate-search

我正在尝试配置使用的Spring Web Service(Spring-WS)应用程序。我正在使用的堆栈如下

Spring 3
Spring-WS
JPA with Hibernate as the provider
JBoss 7.1*

应用程序配置如下:

的persistence.xml

<persistence-unit name="myPersistenceUnit" transaction-type="RESOURCE_LOCAL">
        <provider>org.hibernate.ejb.HibernatePersistence</provider>
        <jta-data-source>java:jboss/datasources/myDataSource</jta-data-source>
        <non-jta-data-source>java:jboss/datasources/myDataSource</non-jta-data-source>
        <class>myPackage.MyClass</class>
        <properties>
            <property name="hibernate.dialect" value="org.hibernate.dialect.MySQLInnoDBDialect"/>           
            <property name="hibernate.connection.autocommit" value="true" />
            <property name="hibernate.hbm2ddl.auto" value="validate"/>
            <property name="hibernate.show_sql" value="false"/>
            <property name="hibernate.cache.use_second_level_cache" value="true"/>
            <property name="hibernate.cache.provider_class" value="net.sf.ehcache.hibernate.SingletonEhCacheProvider"/>
            <property name="hibernate.search.default.directory_provider" value="org.hibernate.search.store.FSDirectoryProvider"/>
            <property name="hibernate.search.default.indexBase" value="./lucene/indexes"/>
            <property name="hibernate.search.default.batch.merge_factor" value="10"/>
            <property name="hibernate.search.default.batch.max_buffered_docs" value="10"/>
        </properties>
    </persistence-unit>

spring.xml - 实体管理器和事务管理器

<context:annotation-config/>

    <context:component-scan base-package="com.mypackage"/>

    <bean class="org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor" />
    <bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />

    <!-- Live database entity and transaction managers -->
    <bean id="entityManagerFactory"
        class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"
        p:dataSource-ref="myDataSource">

        <property name="persistenceUnitName" value="myPersistenceUnit" />
    </bean>

    <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager"
        p:entityManagerFactory-ref="entityManagerFactory" />

    <tx:annotation-driven transaction-manager="transactionManager" />           

Web.xml - 要加载上面的spring.xml文件

<context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>
            /WEB-INF/app-config.xml
            /WEB-INF/spring.xml
            /WEB-INF/spring-datasources.xml
         </param-value>   
     </context-param>

弹簧datasources.xml

<jee:jndi-lookup id="myDataSource" jndi-name="java:jboss/datasources/myDataSource"/>

在我的DAO类中,我对entityManager有以下定义。

@PersistenceContext(type=PersistenceContextType.TRANSACTION, unitName="myPersistenceUnit")
protected EntityManager entityManager;

@Transactional
    public void updateProduct(Product product) {

        //  Save the document to the database
        update(product);
    }

上述更新方法只需调用entityManager.merge(object);

即可调用泛型方法

现在,当我测试上面的内容时,我发现如果我正在从数据库中读取它是有效的,但如果我正在写入数据库(即创建或更新)它不起作用。 如果我尝试写信给我,我会得到以下例外:

javax.persistence.TransactionRequiredException: Executing an update/delete query
   at org.hibernate.ejb.AbstractQueryImpl.executeUpdate(AbstractQueryImpl.java:96)
   at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
   at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
   at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
   at java.lang.reflect.Method.invoke(Method.java:597)
   at org.springframework.orm.jpa.SharedEntityManagerCreator$DeferredQueryInvocationHandler.invoke(SharedEntityManagerCreator.java:310)
   at $Proxy96.executeUpdate(Unknown Source)

我猜测我可以从数据库中读取这意味着事务管理器的配置出了问题。有什么想法吗?

修改

好的,我只是重读上面的内容,发现最后一点有点不对劲。如果我使用JPQL使用entityManager.createQuery发出Update语句,则仅在上面显示的异常发生。如果我尝试使用entityManager.merge()更新实体,则实际上不会发生任何事情。没有返回异常,但行未更新。

2 个答案:

答案 0 :(得分:4)

当你将方法标记为@Transactional时,你正依靠JTA,但是 - 如altanis所述 - 你将持久性单元的事务类型定义为非JTA的RESOURCE_LOCAL。

1)如果您想使用容器管理的事务,那么更改事务类型,在JBoss中将数据源定义为JTA数据源,并将非jta-data-source更改为jta-data-source

2)如果您想管理自己的事务,那么保持persistence.xml和datasource定义不变,但在任何持久化,合并,删除操作之前,通过调用entitymanager.getTransaction()。begin()和在业务单元的末尾,调用entitymanager.getTransaction()。commit()或entitymanager.getTransaction()。rollback()。在这种情况下,由于您自己管理事务,因此在updateProduct()中使用@Transactional启动JTA事务似乎毫无意义。

答案 1 :(得分:1)

尝试在persistence.xml中将transaction-type =“RESOURCE_LOCAL”更改为transaction-type =“JTA”。您还可能需要从persistence.xml中删除元素。