`
narcissusoyf
  • 浏览: 154631 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

“Transaction rolled back because it has been marked as rollback-only”

阅读更多

 

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的顺序啊,哈哈哈。。。

分享到:
评论
2 楼 youlxb2008 2016-03-09  
thank you
讲的很详切
1 楼 canvi 2010-09-16  
  谢谢!收藏

相关推荐

    COS——R.log

    org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only at org.springframework.transaction.support....

    蓝桥杯嵌入式十五届模拟1代码

    蓝桥杯嵌入式十五届模拟1代码

    1-中国各省绿色税收相关数据(2007-2022年)-社科数据.rar

    该数据集收录了2007-2022年中国各省份绿色税收相关数据,涵盖总税收、资源税、环境保护税、城市维护建设税、城镇土地使用税、车船税、耕地占用税、消费税、车辆购置税等指标。绿色税收是以保护环境、合理开发利用自然资源为目的,对特定行为或产品征收的税费,旨在通过经济手段引导企业减排降污,促进资源合理利用和可持续发展。数据包含全国31个省级行政区的年度统计,其中环境保护税自2018年开征后,与资源税等共同构成绿色税收体系,通过"多排多征、少排少征"机制推动企业环保改造。部分数据存在缺失值,但整体可直接用于区域绿色税制比较、环境政策效果评估等研究。

    joblib-0.9.0b4.tar.gz

    该资源为joblib-0.9.0b4.tar.gz,欢迎下载使用哦!

    Android开发根据外框颜色决定文字颜色以及选中颜色demo源码

    Android开发根据标签外框颜色决定字体颜色以及选中后的背景色,这样颜色不同,好看,有点难度,关键在于如何自定义控件

    blue_concrete_powder.png

    blue_concrete_powder

    Android开发仿抖音底部加载进度条demo源码

    Android开发仿抖音底部加载进度条效果,中间往外走的进度条,关键是自定义View

    3D文件硬件开发之3D文件打印,3D生成器,咸蛋超人3D打印文件,5款不同的3D文件免费下载,总有一款适合你,光之铸型:咸蛋超人3D觉醒计划

    【3D文件】硬件开发之3D文件打印,3D生成器,咸蛋超人3D打印文件,5款不同的3D文件免费下载,总有一款适合你,光之铸型:咸蛋超人3D觉醒计划 资源图片查看链接:https://blog.csdn.net/weixin_60324241/article/details/147260889?spm=1011.2415.3001.5331

    water_overlay.png

    j

    基于Springboot框架实现的学生选课管理系统(源码+数据库+万字文档)513

    基于Springboot框架实现的学生选课管理系统,包含一种角色:管理员,系统分为前台登录和后台两大模块,主要功能如下。 【前台登录】: 1. 学生登录:学生可以通过学号和密码登录系统,查看自己的选课情况、成绩等信息。 2. 教师登录:教师可以通过工号和密码登录系统,查看自己的教授课程、学生评价等信息。 【后台模块】: 1. 选课管理:管理员可以管理课程信息,包括添加、编辑、删除课程信息,设置课程容量,调整课程时间表等。 2. 教评管理:管理员可以管理学生对教师的评价,包括查看评价结果、编辑评价内容等。 3. 教评问题管理:管理员可以管理教评中的问题,包括添加、编辑、删除问题,设置问题选项等。 4. 校建管理:管理员可以管理校园建筑信息,包括添加、编辑、删除校建信息,查看校园地图等。 5. 教师管理:管理员可以管理教师信息,包括添加、编辑、删除教师信息,查看教师评价等。 6. 学生管理:管理员可以管理学生信息,包括添加、编辑、删除学生信息,查看学生选课情况等。 7. 课程管理:管理员可以管理课程信息,包括添加、编辑、删除课程信息,查看课程详情等。 8. 成绩管理:管理员可以管理学生成绩,包括录入、编辑、删除成绩,查看学生成绩单等。 9. 校园新闻:管理员可以发布校园新闻,包括添加、编辑、删除新闻,查看新闻详情等。 10. 系统管理:管理员可以管理系统的用户账号和权限,包括添加、编辑、删除用户,设置用户权限等。 二、项目技术 编程语言:Java 数据库:MySQL 项目管理工具:Maven 前端技术:HTML、Jquery、CSS、JavaScript、 后端技术:Spring、SpringMVC、MyBatis 三、运行环境 操作系统:Windows或者macOS JDK版本:最好是JDK1.8,其他版本理论上也可以 开发工具:IDEA、Ecplise

    ParagonHFS+forWin v14.0.24 x64.rar

    WINDOWS系统读取苹果分区的利器,支持HFS+及APFS分区。

    deepseek用例生成+测试工程师智能助手

    1.通过提取docx需求文档有效标题的正文内容、表格及图片OCR结果,结合动态生成的提示词进行功能测试用例生成; 2.通过提取excel、yml或json格式的接口文档,结合提示词生成接口测试用例; 3.用例以json形式预览,支持导出为md、excel、json等格式; 4.安装部署简单,只需python环境即可,安装依赖库后即可运行; 5.需替换脚本里的api-key,模型调用阿里云百炼deepseek-r1的接口;

    美食分类数据集-zip

    数据集介绍: 数据集包含五种美食图片,每种美食类型图片为1000张。

    ### 【物联网操作系统】LiteOS从入门到实战:开发环境搭建、内核解析及网络编程详解、LiteOS简介

    内容概要:本文详细介绍了LiteOS这一轻量级物联网操作系统,涵盖其特点、应用场景、开发环境搭建、内核机制、实战演练及进阶学习。LiteOS由华为开发,专为资源受限设备设计,具备轻量级、高效性、安全性和开放性等特点,适用于智能家居、工业自动化、智能穿戴和智能城市建设等领域。文章逐步讲解了Windows和Linux系统下搭建LiteOS开发环境的具体步骤,包括安装交叉编译器、HiSpark Studio、配置Python环境、下载并配置LiteOS SDK等。深入探讨了LiteOS内核的任务管理和内存管理机制,并通过Hello World程序展示了创建任务、编写代码、编译和烧录的完整流程。最后,介绍了SAL及socket编程,提供了丰富的学习资源,包括官方文档、技术论坛和开源代码库。 适合人群:具备一定编程基础,尤其是对物联网开发感兴趣的开发者,以及希望深入了解嵌入式操作系统原理的技术人员。 使用场景及目标:①学习如何在资源受限的设备上开发高效稳定的应用程序;②掌握LiteOS的任务管理、内存管理等核心机制;③通过实战演练和进阶学习,提高物联网设备的网络通信能力,如使用SAL及socket编程实现设备与服务器之间的TCP通信。 其他说明:本文不仅提供了理论知识,还结合具体代码示例和实际操作步骤,帮助读者更好地理解和应用LiteOS。物联网技术正处于快速发展阶段,掌握LiteOS开发技能将为开发者在智能家居、工业自动化、智能穿戴等领域提供强大的竞争力。

    weathered_copper_door_bottom.png

    j

    有轨装车机机行走机构设计分析.rar

    有轨装车机机行走机构设计分析.rar

    h5py-3.13.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl

    该资源为h5py-3.13.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl,欢迎下载使用哦!

    black_stained_glass.png

    black_stained_glass

    macOS剪辑三剑客Final Cut Pro 11.1附下载

    macOS剪辑三剑客Final Cut Pro 11.1附下载

    酸奶厂总平面设计图.rar

    酸奶厂总平面设计图.rar

Global site tag (gtag.js) - Google Analytics