spring 事务管理之传播行为对回滚的影响(二)

世上无难事,只要肯攀登

不同service中的A方法中调用了B方法。两个方法只有一个用到事务,那么程序的最终结果和代码的执行顺序有很大关系

A方法创建了事务,B方法无事务运行

我曾经以为在A方法中抛出运行时异常,A方法事务被回滚。B方法因为没有纳入事务的管理,所以对B方法毫无影响。但是经过我的实践这个想法是错误的,程序的最终结果其实和代码的执行顺序有很大关系。下面准备实验:

我们给InsertUsers方法加上REQUIRED级别的传播行为,给InsertCuser方法加上NOT_SUPPORTED级别的传播行为。让InsertUsers方法使用事务,InsertCuser方法无事务状态执行。这下肯定有人问:InsertCuser方法不加事务注解不就能保证InsertCuser无事务运行了吗?事实上因为spring的事务传播行为,InsertUsers会将事务传播给InsertCuser,导致两个方法都使用InsertUsers上的事务,所以必须要给InsertCuser方法加上NOT_SUPPORTED级别的传播行为。不信可以看看我的这篇文章 https://www.jianshu.com/p/bc3cbacf9e70 里面有做对应实验

  @Transactional(propagation = Propagation.REQUIRED)
   public void InsertUsers(Users users) {
     。。。
    }

  @Transactional(propagation=Propagation.NOT_SUPPORTED)
   public void InsertCuser(Cuser cuser) {
     。。。
   }

然后分别将下列引发运行时异常的语句放到InsertUsers方法或InsertCuser方法中,观察事务回滚情况

//运行时异常
int i = 1/0;

情况一:

  • 如图在InsertUsers方法中,将抛出异常的代码行放到调用InsertCuser方法之前。程序运行至此出现异常后根本不会再往下执行了,也就是说InsertCuser方法不会得到执行!即使InsertCuser方法没有纳入事务的管理,可以理解为此时它也是执行不成功的~ 查看cuser表数据为空验证了这个结论


    image.png
  • 再来看抛出异常的代码行放到调用InsertCuser方法后,结果就不同了。此时InsertCuser方法会得到执行,因为InsertCuser方法没有纳入到事务管理,它不会回滚。执行程序后查看数据库cuser表数据多出了一行,users表中没有数据


    image.png

情况二:

  • 再将除0异常代码放到没有纳入事务管理的InsertCuser方法方法中,观察事务回滚情况
    把出现异常代码行放到数据库操作之前,程序运行结果是 InsertUsers和InsertCuser方法数据库操作均失效。因为InsertUsers中存在事务,InsertCuser会隐式抛出运行时异常到InsertUsers中,InsertUsers事务得到回滚;而InsertCuser自身数据库操作失效的原因是数据库操作语句那行根本没有得到执行。


    image.png
  • 若把异常代码行放到数据库操作之后,这次InsertUsers同样因为事务遇到异常而回滚,而InsertCuser操作成功执行。


    image.png
A方法无事务运行,B创建事务

我们不给InsertUsers方法使用事务注解,给InsertCuser方法加上REQUIRED级别的传播行为。让InsertUsers无事务状态执行,InsertCuser方法创建事务。InsertUsers之所以没有纳入事务的管理是因为事务不会从被调用者传播到调用者

   public void InsertUsers(Users users)  {
     。。。
   }

   @Transactional(propagation=Propagation.REQUIRED)
   public void InsertCuser(Cuser cuser)  {
     。。。
   }

情况一:

  • 在InsertUsers方法中,把异常代码行放到insert语句之前
    两者都不会插入数据成功,都是因为没有执行到那里去就报异常了


    image.png
  • 在InsertUsers方法中,把异常代码行放到insert语句之后,调用InsertCuser方法之前;程序运行结果是InsertUsers插入数据成功因为InsertUsers没有纳入事务的管理,InsertCuser插入数据失败因为没有执行到那里去


    image.png
  • 在InsertUsers方法中,把异常代码行放到调用InsertCuser方法之后;此时程序的运行结果是:InsertUsers插入数据成功,因为InsertUsers没有纳入事务的管理。InsertCuser插入数据也成功,这里可能很多人不理解,因为这个int i =1/0 引发的运行时异常只在本方法中会影响回滚,也就是说它无法影响已经执行完毕提交的InsertCuser方法中的事务

    image.png

情况二:
无论将int i =1/0放InsertCuser方法块中的哪里,结果都是一样的。
InsertUsers方法插入数据成功,因为它不在事务之内。InsertCuser方法插入数据失败,因为抛出异常事务回滚

大总结

A方法调用B方法;这里的上指的是调用者,下指的是被调用者

  • 异常总是向上抛出,B方法会将异常抛出到A方法
  • 事务总是伴随着异常的抛出而回滚,而且B方法事务如果进行抛出会影响到A方法的事务。 方法中如果将异常抛出给上层调用者方法,那么当前方法的事务会回滚 像除0异常这种运行时异常是隐式抛出的不用管、受检异常就必须显式抛出,本方法的事务才会回滚
  • 事务总是向下传播,A方法中有事务,B方法中没事务。A方法可能会将事务传播给B方法
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容