spring的声明事务提供了强大功能,让我们把业务关注和非业务关注的东西又分离开了。好东西的使用,总是需要有代价的。使用声明事务的时候,一个不小心经常会碰到“Transaction rolled back because it has been marked as rollback-only”这个异常。有时候又常常会纳闷,"我已经try-catch了,为什么还这样呢?"
<!-- 0 placeHolder -->
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<value>files/pro.properties</value>
</list>
</property>
</bean>
<!-- 1 dataSource -->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
<property name="url" value="${jdbc.mysql.url}"></property>
<property name="username" value="${jdbc.username}"></property>
<property name="password" value="${jdbc.userpassword}"></property>
</bean>
<!-- 2 jdbcTemplate -->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 3 BaseDao -->
<bean id="baseDao" class="transaction.dao.BaseDao" abstract="true">
<property name="jdbcTemplate" ref="jdbcTemplate" />
</bean>
<bean id="aDao" class="transaction.dao.Adao" parent="baseDao">
</bean>
<bean id="bDao" class="transaction.dao.Bdao" parent="baseDao">
</bean>
<!-- 4 service -->
<bean id="aBo" class="transaction.bo.AboImpl">
<property name="aDao" ref="aDao" />
<property name="bBo" ref="bBo" />
</bean>
<bean id="bBo" class="transaction.bo.BboImpl">
<property name="bDao" ref="bDao" />
</bean>
<!-- 5 transaction -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<bean id="transactionInterceptor1" class="org.springframework.transaction.interceptor.TransactionInterceptor" >
<property name="transactionManager" ref="transactionManager"></property>
<property name="transactionAttributes">
<props>
<prop key="*">PROPAGATION_REQUIRED</prop>
</props>
</property>
</bean>
<bean id="autoProxy1" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<!--
<value>transactionInterceptor2</value>
-->
<value>transactionInterceptor1</value>
</list>
</property>
</bean>
这里的声明事务是作用于所有以Bo为后缀的bean的所有方法上,使用REQUIRED传播方式。
public int insertA(A a)
{
aDao.insertA(a);
B b = new B();
b.setName("bbb");
try
{
bBo.insertB(b);
}
catch(Exception e)
{
System.out.println("aaa");
}
return 0;
}
这里,insertA 开始一个事务,调用aDao.insertA(a)[一个简单的数据库操作],然后调用 bBo.insertB(b)[bo调dao,dao直接抛异常]。bBo的insertB方法,也要开始一个事务,但是这里的传播机制是REQUIRED。OK,和insertA 的事务合二为一吧。因为bBo.insertB(b)会抛异常出来,这里try-catch下,希望aDao.insertA(a)的操作能够成功。
但是现实总是残酷的,这里会有一个大大的 “Transaction rolled back because it has been marked as rollback-only” ,结果你会发现aDao.insertA(a)的操作也没有成功。
try-catch不起作用的原因简单的说就是,try-catch的不是地方,你认为你的try-catch是最接近异常抛出点了,是第一个处理的handler了。实际上,spring在更早一步就try-catch 住了,同时还设置了一些标志位,再把catch住的异常往外抛。这个时候才是我们的try-catch。而"Transaction rolled back because it has been marked as rollback-only"就是因为事务在提交的时候,发现标志位已经被设置了,不应该去提交了,然后吭哧吭哧的回滚调,再提示你已经被设置成rollback-only了。
原因是既然如此,那么在不改变代码的情况下,依靠配置能否解决这个问题呢?使用PROPAGATION_REQUIRES_NEW吧。对于bBo.insertB(b)开个新的事务,如果失败了就回滚调,不影响外面的insertA不就OK了。最简单的情况就是在transactionInterceptor1前面,再加个拦截器transactionInterceptor2,该拦截器只针对insertB的事务属性进行修改。
<bean id="autoProxy1" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor2</value>
<value>transactionInterceptor1</value>
</list>
</property>
</bean>
<bean id="transactionInterceptor2" class="org.springframework.transaction.interceptor.TransactionInterceptor" >
<property name="transactionManager" ref="transactionManager"></property>
<property name="transactionAttributes">
<props>
<prop key="insertB">PROPAGATION_REQUIRES_NEW</prop>
</props>
</property>
</bean>
注意interceptorNames里面元素的位置。先要使用transactionInterceptor2,再使用transactionInterceptor1.因为调用insertB的时候,transactionInterceptor2先开了一个新事务,而后transactionInterceptor1融合进这个事务。如果这2个拦截器的顺序颠倒的话,那么还是会出现“Transaction rolled back because it has been marked as rollback-only”。因为,transactionInterceptor2生成事务回滚以后,还是会把ex抛给transactionInterceptor1。这个时候,transactionInterceptor1的事务和insertA的事务是同一个。transactionInterceptor1,把标志设置好,等到insertA真的结束的时候,因为异常被我们的try-catch捕获了,spring就会发现需要提交的事务具有一个已经被标记号的rollback。所以就又抛出来了。
但是如果系统有很多遗留的因素导致你不敢盲目的修改配置文件的话(比如事务的poincut),那么我们就再加一个事务proxy就OK了。
<bean id="autoProxy2" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor2</value>
<!--
<value>transactionInterceptor1</value>
-->
</list>
</property>
</bean>
<bean id="autoProxy1" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor1</value>
<!--
<value>transactionInterceptor2</value>
-->
</list>
</property>
</bean>
如上的配置还是会带来悲剧的“Transaction rolled back because it has been marked as rollback-only”。
但是如果我们把 autoProxy2 放到 autoProxy1 或者给自动代理加上顺序的话。。。结果就是喜剧了。。
<bean id="autoProxy1" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor1</value>
<!--
<value>transactionInterceptor2</value>
-->
</list>
</property>
</bean>
<bean id="autoProxy2" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor2</value>
<!--
<value>transactionInterceptor1</value>
-->
</list>
</property>
</bean>
造成这个原因是由使用了2个代理的顺序导致的。
在做自动代理的时候,spring会按照postBeanProcessor bean声明的顺序(如果没有设置顺序的话),来依次处理bean。如果autoProxy2 在 autoProxy1 之前,这样transactionInterceptor2 就会更加贴近insertB的调用,其效果就像
<bean id="autoProxy1" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
<property name="beanNames">
<list>
<value>*Bo</value>
</list>
</property>
<property name="interceptorNames">
<list>
<value>transactionInterceptor1</value>
<value>transactionInterceptor2</value>
</list>
</property>
</bean>
的配置。
看来~~~ spring 还是要注意bean的顺序啊,哈哈哈。。。
分享到:
相关推荐
org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only at org.springframework.transaction.support....
在IT领域,SpringBoot、Activiti和Oracle是三个非常重要的技术组件,它们分别代表了现代企业级应用开发的三大支柱:轻量级框架、工作流引擎和关系型数据库。本项目将这三者进行了深度整合,构建了一个高效、便捷的...
本文将围绕“asec-w04-derived-unique-token-per-transaction - APT”这一主题,深入探讨工控安全、数据泄露、勒索软件、大数据以及高级持续性威胁(APT)等关键知识点,同时也会提及安全集成、企业安全策略以及零...
赠送jar包:javax.transaction-api-1.2.jar; 赠送原API文档:javax.transaction-api-1.2-javadoc.jar; 赠送源代码:javax.transaction-api-1.2-sources.jar; 赠送Maven依赖信息文件:javax.transaction-api-1.2....
Could not roll back Hibernate transaction; nested exception is org.hibernate.TransactionException: JDBC rollback failed 这表明Hibernate事务回滚操作失败,导致事务不能正确回滚。 二、问题原因 该问题的...
Intent locks improve performance because SQL Server examines intent locks only at the table level to determine whether a transaction can safely acquire a lock on that table. This removes the ...
赠送jar包:javax.transaction-api-1.2.jar; 赠送原API文档:javax.transaction-api-1.2-javadoc.jar; 赠送源代码:javax.transaction-api-1.2-sources.jar; 赠送Maven依赖信息文件:javax.transaction-api-1.2....
transaction-processing-concepts-and-techniques
《JBoss AS 7.1.0.Final:企业级应用服务器详解》 JBoss AS,全称为"Java EE Application Server",是Red Hat公司开发的一款开源、免费的应用服务器,它基于Java平台的企业级服务(Java Platform, Enterprise ...
jboss-transaction-api
- A hang on startup has been corrected. A 2 minute timeout has been added to the collection of system information. - Video playback, Hard disk and CD/DVD test 'no operations' error reporting ...
<tx:advice id="txAdvice" transaction-manager="myJtaManager"> *" rollback-for="Exception"/> *" rollback-for="Exception"/> *" rollback-for="Exception"/> *" read-only="true" rollback-for=...
<artifactId>tcc-transaction-api 版本号 <groupId>com.alipay.sofa</groupId> <artifactId>tcc-transaction-core 版本号 <groupId>com.alipay.sofa</groupId> <artifactId>tcc-transaction-spring ...
jboss-transaction-api_1.1_spec-1.0.1.Final.jar包 描述: There is a new version for this artifact
分布式事务框架LCN支持springcloud 2.0.5 ,transaction-springcloud-4.1.2.jar
在这个名为"TCC-Transaction分布式事务DEMO"的项目中,我们将深入探讨TCC模式以及如何在基于Mysql数据库的环境中实现它。 **TCC模式详解** TCC(Try-Confirm-Cancel)是一种补偿型的分布式事务处理模式,主要由三...
语 句 功 能 数据操作 SELECT --从数据库表中检索数据...SET TRANSACTION --定义当前事务数据访问特征 --程序化 SQL DECLARE --为查询设定游标 EXPLAN --为查询描述数据访问计划 OPEN --检索查询结果打开一个游标
本文将深入探讨“C#中调用Transaction”的相关知识点,包括事务的基本概念、事务的特性、如何在C#代码中创建和管理事务,以及与数据库交互时事务的应用。 ### 1. 事务基本概念 事务是数据库操作的一组逻辑单元,它...