“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了,为什么还这样呢?"

 

Xml代码  收藏代码
  1. <!-- 0 placeHolder -->  
  2. <bean  class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">  
  3.     <property name="locations">  
  4.         <list>  
  5.             <value>files/pro.properties</value>  
  6.         </list>  
  7.     </property>  
  8. </bean>  
  9.   
  10. <!-- 1 dataSource -->  
  11. <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">  
  12.     <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>  
  13.     <property name="url" value="${jdbc.mysql.url}"></property>  
  14.     <property name="username" value="${jdbc.username}"></property>  
  15.     <property name="password" value="${jdbc.userpassword}"></property>  
  16. </bean>  
  17.   
  18. <!-- 2 jdbcTemplate -->  
  19. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">  
  20.     <property name="dataSource" ref="dataSource"></property>  
  21. </bean>  
  22.   
  23. <!-- 3 BaseDao -->  
  24. <bean id="baseDao" class="transaction.dao.BaseDao" abstract="true">  
  25.     <property name="jdbcTemplate" ref="jdbcTemplate" />  
  26. </bean>  
  27.   
  28. <bean id="aDao" class="transaction.dao.Adao" parent="baseDao">  
  29. </bean>  
  30.   
  31. <bean id="bDao" class="transaction.dao.Bdao" parent="baseDao">   
  32. </bean>  
  33.   
  34. <!-- 4 service -->  
  35. <bean id="aBo" class="transaction.bo.AboImpl">  
  36.     <property name="aDao" ref="aDao" />  
  37.     <property name="bBo" ref="bBo" />  
  38. </bean>  
  39.   
  40. <bean id="bBo" class="transaction.bo.BboImpl">  
  41.     <property name="bDao" ref="bDao" />  
  42. </bean>  
  43.   
  44. <!-- 5 transaction -->  
  45. <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">  
  46.     <property name="dataSource" ref="dataSource" />  
  47. </bean>  
  48.   
  49. <bean id="transactionInterceptor1" class="org.springframework.transaction.interceptor.TransactionInterceptor" >  
  50.     <property name="transactionManager" ref="transactionManager"></property>  
  51.     <property name="transactionAttributes">  
  52.         <props>  
  53.             <prop key="*">PROPAGATION_REQUIRED</prop>  
  54.         </props>    
  55.     </property>  
  56. </bean>  
  57.   
  58. <bean id="autoProxy1"  class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  59.     <property name="beanNames">  
  60.         <list>  
  61.             <value>*Bo</value>  
  62.         </list>  
  63.     </property>  
  64.     <property name="interceptorNames">  
  65.         <list>  
  66.         <!--   
  67.             <value>transactionInterceptor2</value> 
  68.             -->  
  69.             <value>transactionInterceptor1</value>    
  70.         </list>  
  71.     </property>  
  72. </bean>  

 这里的声明事务是作用于所有以Bo为后缀的bean的所有方法上,使用REQUIRED传播方式。

Java代码  收藏代码
  1. public int insertA(A a)   
  2. {  
  3.     aDao.insertA(a);  
  4.       
  5.     B b = new B();  
  6.     b.setName("bbb");  
  7.     try  
  8.     {  
  9.         bBo.insertB(b);  
  10.     }  
  11.     catch(Exception e)  
  12.     {  
  13.         System.out.println("aaa");  
  14.     }  
  15.     return 0;  
  16. }  

 这里,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的事 务属性进行修改。

Xml代码  收藏代码
  1. <bean id="autoProxy1"  class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  2.     <property name="beanNames">  
  3.         <list>  
  4.             <value>*Bo</value>  
  5.         </list>  
  6.     </property>  
  7.     <property name="interceptorNames">  
  8.         <list>  
  9.            
  10.             <value>transactionInterceptor2</value>  
  11.               
  12.             <value>transactionInterceptor1</value>    
  13.         </list>  
  14.     </property>  
  15. </bean>  
  16.   
  17.   
  18. <bean id="transactionInterceptor2" class="org.springframework.transaction.interceptor.TransactionInterceptor" >  
  19.     <property name="transactionManager" ref="transactionManager"></property>  
  20.     <property name="transactionAttributes">  
  21.         <props>  
  22.             <prop key="insertB">PROPAGATION_REQUIRES_NEW</prop>  
  23.         </props>    
  24.     </property>  
  25. </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了。

Xml代码  收藏代码
  1. <bean id="autoProxy2" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  2.         <property name="beanNames">  
  3.             <list>  
  4.                 <value>*Bo</value>  
  5.             </list>  
  6.         </property>  
  7.         <property name="interceptorNames">  
  8.             <list>  
  9.                 <value>transactionInterceptor2</value>  
  10.                 <!--   
  11.                 <value>transactionInterceptor1</value>   
  12.                 -->  
  13.             </list>  
  14.         </property>  
  15.     </bean>  
  16.       
  17.     <bean id="autoProxy1"  class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  18.         <property name="beanNames">  
  19.             <list>  
  20.                 <value>*Bo</value>  
  21.             </list>  
  22.         </property>  
  23.         <property name="interceptorNames">  
  24.             <list>  
  25.             <value>transactionInterceptor1</value>    
  26.             <!--   
  27.             <value>transactionInterceptor2</value>       
  28.             -->        
  29.             </list>  
  30.         </property>  
  31.     </bean>  

如上的配置还是会带来悲剧的“Transaction rolled back because it has been marked as rollback-only”。

 

但是如果我们把 autoProxy2 放到 autoProxy1 或者给自动代理加上顺序的话。。。结果就是喜剧了。。

Xml代码  收藏代码
  1. <bean id="autoProxy1"  class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  2.         <property name="beanNames">  
  3.             <list>  
  4.                 <value>*Bo</value>  
  5.             </list>  
  6.         </property>  
  7.         <property name="interceptorNames">  
  8.             <list>  
  9.             <value>transactionInterceptor1</value>    
  10.             <!--   
  11.             <value>transactionInterceptor2</value>       
  12.             -->        
  13.             </list>  
  14.         </property>  
  15.     </bean>  
  16.       
  17.       
  18.     <bean id="autoProxy2" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  19.         <property name="beanNames">  
  20.             <list>  
  21.                 <value>*Bo</value>  
  22.             </list>  
  23.         </property>  
  24.         <property name="interceptorNames">  
  25.             <list>  
  26.                 <value>transactionInterceptor2</value>  
  27.                 <!--   
  28.                 <value>transactionInterceptor1</value>   
  29.                 -->  
  30.             </list>  
  31.         </property>  
  32.     </bean>  

 

 

造成这个原因是由使用了2个代理的顺序导致的。

在做自动代理的时候,spring会按照postBeanProcessor bean声明的顺序(如果没有设置顺序的话),来依次处理bean。如果autoProxy2 在 autoProxy1 之前,这样transactionInterceptor2 就会更加贴近insertB的调用,其效果就像

Xml代码  收藏代码
  1. <bean id="autoProxy1"  class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">  
  2.         <property name="beanNames">  
  3.             <list>  
  4.                 <value>*Bo</value>  
  5.             </list>  
  6.         </property>  
  7.         <property name="interceptorNames">  
  8.             <list>  
  9.             <value>transactionInterceptor1</value>    
  10.            
  11.             <value>transactionInterceptor2</value>        
  12.                   
  13.             </list>  
  14.         </property>  
  15.     </bean>  

 的配置。

 

 

看来~~~ spring 还是要注意bean的顺序啊,哈哈哈。。。

转载于:https://www.cnblogs.com/langtianya/p/4962784.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/431124.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

c++获取时间戳_时间简史

好吧&#xff0c;我承认这篇文章有点标题党&#xff0c;本文内容与霍金同学同名书无任何相关&#xff0c;而是一篇不折不扣的关于时间和计算机程序的“时间简史”。时间是什么&#xff1f;数学家说时间是第四维度&#xff0c;不过显然计算机不这么看&#xff0c; 在计算机的世界…

[iOS] 建立与使用Framework

[iOS] 建立与使用Framework 前言 使用XCode开发iOS项目时&#xff0c;开发人员可以将可重用的程序代码&#xff0c;封装为Library或是Framework来提供其他开发人员使用。这两种封装方式在使用的时候&#xff1a;Library需要将.a封装档与所有公开的.h档提供给使用者加入项目&…

a标签点击事件_html常用标签

一、a标签作用&#xff1a;1、跳转到外部页面&#xff1b;2、跳转到内部锚点&#xff1b;3、跳转到邮箱或电话。属性&#xff1a;href&#xff08;hyperreference&#xff09;&#xff1a;超级引用、超级链接1、a的href的取值&#xff1a;<//http://google.com&#xff1a;无…

mysql导出表_mysql导出表的3种方式

文章转载自 &#xff1a;https://blog.csdn.net/jbb0403/article/details/26359573navicat导出表一共有三中用法&#xff1a;第一种&#xff1a;数据库上右键—>"转储SQL文件"&#xff0c;如图&#xff1a;"转储文件"是把整个数据库表全部导出&#xff…

网络资产管理系统_固定资产管理系统的常用操作

固定资产管理系统作为一个办公软件&#xff0c;能够帮助企业高效管理固定资产的日常事务。随着信息技术的发展和普及&#xff0c;越来越多的企业都开始使用固定资产管理系统&#xff0c;可是对于首次使用或者没有接触过固定资产管理系统的新手企业来说&#xff0c;对于系统的了…

mac mysql my.cnf_mac 增加 my.cnf

mac 增加 my.cnf发布时间&#xff1a;2018-04-25作者&#xff1a;laosun阅读(8133)mysql下没有my.cnf&#xff0c;如果想要修改一些配置还挺懵逼的&#xff0c;网上找了一些资料&#xff0c;经过博主亲测没有问题&#xff0c;发篇文章一为共享&#xff0c;二为做个备忘。博主使…

python取前三位_Python 实现取多维数组第n维的前几位

Python 实现取多维数组第n维的前几位 现在我们有一个shape为(7352, 9, 128, 1)的numpy数组。 想要取出第2维的前三个数据&#xff0c;构成新数组(7352, 3, 128, 1) 我的思想是&#xff1a;将第2维数据转置&#xff08;transpose&#xff09;到第一维&#xff0c;再用切片&#…

solr 7 mysql导入_solr 7.7.0 windows 导入mysql数据库数据

接上一篇 准备导入数据首先修改hello/cong目录下的solrconfig.xml文件&#xff0c;添加如下节点&#xff1a;data-config.xml新建一个data-config.xml文件&#xff0c;与solrconfig.xml同一个目录下 添加数据库链接信息DataSource&#xff1a;数据库连接信息Entity&#xff1a;…

access vba代码大全_VBA 实践指南 -- VBA连接各种数据库

ADO 简介ADO (ActiveX Data Objects&#xff0c;ActiveX数据对象&#xff09;是Microsoft提出的应用程序接口&#xff08;API&#xff09;用以实现访问关系或非关系数据库中的数据。例如&#xff0c;如果您希望编写应用程序从DB2或Oracle数据库中向网页提供数据&#xff0c;可以…

sql date类型_共享单车数据分析的SQL数据库设计

SQL&#xff0c;发音为“ sequel”(或SQL&#xff0c;如果愿意的话)&#xff0c;是数据科学家的重要工具。实际上&#xff0c;它可以说是获取数据工作中最重要的语言。在共享单车数据分析的SQL设计中&#xff0c;我们将从入门者的角度深入研究SQL基础知识&#xff0c;以使您入门…

python header头部请求_python3爬虫怎样构建请求header

写一个爬虫首先就是学会设置请求头header&#xff0c;这样才可以伪装成浏览器。下面小编我就来给大家简单分析一下python3怎样构建一个爬虫的请求头header。 1、python3跟2有了细微差别&#xff0c;所以我们先要引入request&#xff0c;python2没有这个request哦。然后复制网址…

mysql配置文件结构_MariaDB/MySQL配置文件my.cnf解读

MariaDB/MySQL的默认设置性能非常差&#xff0c;仅仅起一个功能测试的作用&#xff0c;不能用在生产环境中&#xff0c;因此要对一些参数进行调整优化。当然&#xff0c;对配置文件各参数的调整需要根据实际环境&#xff0c;不同时期不同数量级的数据进行性能优化。MySQL/Maria…

python各版本区别_关于python中不同版本的print区别

最近有时候会看看python的一些东西。也看了一些入门教程&#xff0c;但是发现一个问题是&#xff0c;在用print的时候老是报错&#xff1a; 报错如下&#xff1a;提示print a这句有语法问题。但是看的教程里面都有这样写的&#xff0c;没有问题啊。 忽然想到以前同学说过python…

stc8g1k08程序范例_通过WiFi对STC单片机程序下载和调试

简介本文设计了基于WiFi-UART的STC单片机下载调试模块&#xff0c;这样便于对嵌入式系统进行远程调试开发&#xff0c;特别是在一些具有移动功能的平台&#xff0c;强电平台以及需要实时监视和修改程序的应用场合。本文相关的下载资源&#xff1a;WiFi模块下载STC程序设计文件[…

echo off备份mysql_windows下备份Linux服务器上的MySQL

【更新】因为windows存在 10点前后生成的时间格式不同&#xff0c;导致无法正确生成sql文件。导致上次的备份文件无法生成研究后对旧内容进行修改echo offforfiles /p "D:\ERP_MySQLbak" /m mysql_backup_*.sql -d -2 /c "cmd /c del /f path"set/a a%time…

python 3d重建_3D点云重建原理及Pytorch实现

3D点云重建原理及Pytorch实现 Pytorch: Learning Efficient Point Cloud Generation for Dense 3D Object Reconstruction 一种Pytorch实现方法&#xff1a;学习高效的点云生成方法用于稠密三维物体重建 一&#xff0e;论文概要 传统的三维对象生成建模方法是利用三维卷积运算的…

Java进阶04 RTTI

链接地址&#xff1a;http://www.cnblogs.com/vamei/archive/2013/04/14/3013985.html 作者&#xff1a;Vamei 出处&#xff1a;http://www.cnblogs.com/vamei 欢迎转载&#xff0c;也请保留这段声明。谢谢&#xff01; 运行时类型识别(RTTI, Run-Time Type Identification)是J…

springdatajpa命名规则_Spring Boot 之Spring data JPA简介

PA的全称是Java Persistence API (JPA)&#xff0c;他是一个存储API的标准&#xff0c;而Spring data JPA就是对JPA的一种实现&#xff0c;可以让我们方便的对数据进行存取。按照约定好的方法命名规则写dao层接口&#xff0c;从而在不实现接口的情况下&#xff0c;实现对数据库…

PHPCMS 模板的设置

编制好模板后&#xff0c; 一、设置config.php<?php return array (name > 现代的蓝红色模板,author > dazhaihui,dirname > simpleblue,homepage > http://www.phpcms.cn/,version > 1.0,disable > 0,file_explan >array (templates|simpleblue|cont…

奖学金设计mysql_基于JAVA的奖学金评定系统设计与实现(MySQL)(含录像)

基于JAVA的奖学金评定系统设计与实现(MySQL)(含录像)(任务书,开题报告,外文翻译,毕业论文10000字,程序代码,MySQL数据库,答辩PPT,答辩视频录像)摘 要本文论述了奖学金评定系统从分析到实现的整个过程&#xff0c;说明了系统实现的基本思路&#xff0c;介绍系统不同的功能模块以…