手把手利用实战代码带你读懂Spring的事务传播行为
创始人
2025-05-29 23:32:14
0

概念

首先简单了解一下 Spring 中事务传播行为是什么?听起来很高端,但是真正用起来的时候,稍有不慎,就会让自己陷入困境之中,所以在使用之前,我们必须要十分耐心认真的学习它。

从名字理解起来,事务传播行为,既然为传播就肯定发生在两个实体之间,否则单个实体又如何发生行为呢。通俗点讲就是“一个巴掌拍不响”。下面进入正规话题。

事务传播行为主要用来描述由某一个事务传播行为修饰的方法被嵌套进另一个方法的事务中,该事务如何传播。这个概述可能不好理解,换句话就是当一个事务方法被另一个事务方法调用时,这个事务方法应该如何进行。

下面用代码+文字说明解释上面的概念。

@Transaction(Propagation=XXX)
public void methodA(){methodB();//doSomething}@Transaction(Propagation=XXX)public void methodB(){//doSomething}

methodA 事务方法调用 methodB 事务方法时,methodB 是继续在调用者 methodA 的事务中运行呢,还是为自己开启一个新事务运行,这就是由 methodB 的事务传播行为决定的。

注意:methodA 和 methodB 都加了事务。methodA()也可以不用开启事务,某一个事务传播行为修饰的方法并不是必须要在开启事务的外围方法中调用

Spring 中七种事务传播行为

通过上面伪代码加文字解释了解到事务传播行为的相关概念,下面就要学习事务传播行为的类型和运行机制。

Spring 中七种事务传播行为

验证

Propagation_Required

调用者方法不存在事务传播行为

「1.调用者方法内部存在异常时,被调用者方法均存在事务,那么结果如何呢?」

@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertClass(ClassDo classDo) {classMapper.insertClass(classDo);System.out.println("----------------------->Class插入成功!");
}

单元测试

// 单元测试
@SpringBootTest
@RunWith(SpringRunner.class)
public class PropagationTest {private final static StudentDo studentDo = new StudentDo();private final static ClassDo classDo = new ClassDo();static {studentDo.setClassId(1);studentDo.setStudentName("student1");studentDo.setAddress("测试");classDo.setClassName("class_1");classDo.setClassNo("Class01");}@Autowiredprivate StudentService studentService;@Autowiredprivate ClassService classService;@Testpublic void insertTest() {studentService.insertStudent(studentDo);classService.insertClass(classDo);}
}

Propagation_Required事务传播机制

结果:两条数据均被插入数据库。由于外部方法并没有开启事务,所以内部方法均在自己的事务提交或者回滚,因此外部方法中存在异常,内部方法事务不会回滚。

「2.被调用者均存在事务,而在被调用者中存在异常,那么结果如何?」

@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}//此方法中抛出异常
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertClassByException(ClassDo classDo) throws CustomException {classMapper.insertClass(classDo);throw new CustomException();
}

单元测试代码

//单元测试代码
private final static StudentDo studentDo = new StudentDo();
private final static ClassDo classDo = new ClassDo();
static {studentDo.setClassId(2);studentDo.setStudentName("student2");studentDo.setAddress("测试2");classDo.setClassName("class_2");classDo.setClassNo("Class02");
}@Test
public void insertExceptionTest() throws CustomException {studentService.insertStudent(studentDo);classService.insertClassByException(classDo);
}

Propagation_Required

结果:第一数据成功插入,第二条数据因异常存在,事务回滚。内部方法均在各个的事务中运行,class 事务回滚,student 数据不会受到影响。

                       

结合 1 和 2 我们可以得出结论:

  1. 通过这两个方法我们证明了在外围方法未开启事务的情况下Propagation_Required修饰的内部方法会新开启自己的事务,且开启的事务相互独立,互不干扰。

  2. 调用者开启事务传播行为。

内部方法同上

//单元测试方法
@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertInnerExceptionThrowsTest() throws CustomException {studentService.insertStudent(studentDo);classService.insertClassByException(classDo);
}

结果:内部方法虽然存在事务传播行为,但是外部方法也存在事务且使用Propagation.REQUIRED修饰,所有内部方法不会新建事务,直接运行在当前事务中,所以 student、class 均会被回滚。

  1. 调用者开启事务传播行为,但是捕获内部方法异常。

/*** 内部方法发生异常情况,外部方法即使捕获处理该异常,依然数据会被回滚*/
@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertInnerExceptionTest() {studentService.insertStudent(studentDo);try {classService.insertClassByException(classDo);} catch (CustomException e) {e.printStackTrace();}
}

结果:外围方法开启事务,内部方法加入外围方法事务,内部方法抛出异常回滚,即使方法被 catch 不被外围方法感知,整个事务依然回滚。同 2 一样,调用者方法执行操作和被调用者中的方法操作结果均被回滚。

Propagation_Supports

@Transactional(propagation = Propagation.SUPPORTS, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertSupportsTest() {studentService.insertStudent(studentDo);
}

解释:如果单纯的调用insertStudent()方法,则以非事务执行,即使后面存在异常情况,执行操作结果不会触发事务回滚机制。当调用insertSupportsTest()方法时,该方法以 REQUIRED 修饰,则会新建一个事务,内部调用insertStudent()方法,所以insertStudent()会加入到当前事务中执行。

Propagation_Mandatory

@Transactional(propagation = Propagation.MANDATORY, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertSupportsTest() {studentService.insertStudent(studentDo);
}

Propagation_Mandatory

解释结果:MANDATORY 表示被修饰的方法必须在事务中运行。当单独调用 insertStudent 时,因为当前没有一个活动的事务,则会抛出异常。

throw new IllegalTransactionStateException(“Transaction propagation ‘mandatory’ but no existing transaction found”);

当调用 insertSupportsTest 时,insertStudent 则加入到 insertSupportsTest 的事务中,事务地执行。

Propagation_Required_New

表示被修饰的方法必须运行在它自己的事务中。一个新的事务会被启动。如果调用者存在当前事务,则在该方法执行期间,当前事务会被挂起。

private final static StudentDo studentDo = new StudentDo();
private final static ClassDo classDo = new ClassDo();
static {studentDo.setClassId(2);studentDo.setStudentName("requireNew");studentDo.setAddress("requireNew");
}
@Transactional(propagation = Propagation.REQUIRES_NEW, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}
@Transactional(propagation = Propagation.REQUIRES_NEW, rollbackFor = Exception.class)
public void insertClassByException(ClassDo classDo) throws CustomException {classMapper.insertClass(classDo);throw new CustomException();
}@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertInnerExceptionTest() {studentService.insertStudent(studentDo);try {classService.insertClassByException(classDo);} catch (CustomException e) {e.printStackTrace();}
}

结果解析:insertStudent(),insertClassByException() 方法执行时,外部方法事务被挂起,内部方法会新建事务,直至该方法执行结束,恢复外部方法事务执行。两者之间事务存在隔离性,insertClassByException() 方法遇到异常,触发事务回滚机制,但 insertStudent() 执行结果并受到影响。

如图所示:

触发事务回滚机制

Propagation_Required_New

Propagation_Not_Supported

表示被修饰的方法不应该运行在事务中。如果调用者存在当前事务,则该方法运行期间,当前事务将被挂起。

private final static ClassDo classDo = new ClassDo();
static {classDo.setClassName("notSupport");classDo.setClassNo("notSupport");
}
@Transactional(propagation = Propagation.NOT_SUPPORTED, rollbackFor = Exception.class)
public void insertClassByException(ClassDo classDo) throws CustomException {classMapper.insertClass(classDo);throw new CustomException();
}
@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertInnerExceptionTest() {try {classService.insertClassByException(classDo);} catch (CustomException e) {e.printStackTrace();}
}

结果解释:即使外部方法开启事务,但是 insertClassByException() 执行,当前事务会挂起,not_support 以非事务方式运行,所以即使遇到异常情况,执行结果也不会触发回滚。

Propagation_Not_Supported

Propagation_Never

表示被修饰的方法不应该运行事务上下文中。如果调用者或者该方法中存在一个事务正在运行,则会抛出异常。

@Transactional(propagation = Propagation.NEVER, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertInnerExceptionTest() {studentService.insertStudent(studentDo);
}

结果如图:

Propagation_Nested

  • 表示当前方法已经存在一个事务,那么该方法将会在嵌套事务中运行。

  • 嵌套的事务可以独立与当前事务进行单独地提交或者回滚。

  • 如果当前事务不存在,那么其行为与 Propagation_Required 一样。

  • 嵌套事务的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚。

「1.外部未开启事务时,内部方法则新建事务执行」

private final static StudentDo studentDo = new StudentDo();
private final static ClassDo classDo = new ClassDo();
static {studentDo.setClassId(2);studentDo.setStudentName("NESTED");studentDo.setAddress("NESTED");classDo.setClassName("NESTED");classDo.setClassNo("NESTED");
}
@Test
public void insertTest() {studentService.insertStudent(studentDo);classService.insertClass(classDo);throw new RuntimeException();}
@Transactional(propagation = Propagation.NESTED, rollbackFor = Exception.class)
public void insertStudent(StudentDo studentDo) {studentMapper.insertStudent(studentDo);System.out.println("----------------------->Student插入成功!");
}
@Transactional(propagation = Propagation.NESTED, rollbackFor = Exception.class)
public void insertClass(ClassDo classDo) {classMapper.insertClass(classDo);System.out.println("----------------------->Class插入成功!");
}

结果:

Propagation_Nested

Propagation_Nested

「2.外部方法开启事务:」

  • 如果外部方法发生异常,则内部事务一起发生回滚操作;

  • 如果外部无异常情况,内部被调用方法存在异常情况,则内部方法独立回滚

//单测代码
private final static StudentDo studentDo = new StudentDo();
private final static ClassDo classDo = new ClassDo();
static {studentDo.setClassId(2);studentDo.setStudentName("NESTED_InnerException");studentDo.setAddress("NESTED_InnerException");classDo.setClassName("NESTED_InnerException");classDo.setClassNo("NESTED_InnerException");
}
@Test
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void insertInnerExceptionThrowsTest() throws CustomException {studentMapper.insertStudent(studentDo);classService.insertClassByException(classDo);
}
//NESTED事务传播行为
@Transactional(propagation = Propagation.NESTED, rollbackFor = Exception.class)
public void insertClassByException(ClassDo classDo) throws CustomException {classMapper.insertClass(classDo);throw new RuntimeException();
}

相关内容

热门资讯

阿联酋最大银行及另两家中东银行... 观点网讯:5月8日,路透社报道指,阿联酋最大银行第一阿布扎比银行(First Abu Dhabi B...
深圳239亿地王易主,再造万象... 2017年,世茂集团豪掷239.43亿元拿下世茂深港国际中心地块,曾规划建筑高度达700米的深圳第一...
蔚来在安庆成立新能源科技公司 ... 天眼查App显示,近日,安庆蔚来新能源科技有限公司成立,法定代表人为姚蒀,注册资本500万人民币,经...
美国牛肉商期盼峰会重启对华出口 据路透社5月8日报道,美国牛肉生产商正期待特朗普与中国于5月14日至15日的峰会推动对华出口许可恢复...
创业板首家未盈利企业,市值突破... 5月8日,大普微总市值正式突破2000亿元大关。截至午间收盘,大普微涨14.07%,报493.1元/...
招商证券:董事长霍达因工作变动... 招商证券公告,公司董事长霍达因工作变动申请辞去董事长、执行董事等全部职务,辞任自辞呈送达董事会之日生...
原创 中... 【阅读须知】本文所引用的所有信息和数据,均为作者通过查阅官方资料与网络公开数据整理、分析而成,旨在为...
原创 从... 2026年5月5日,中国商务部发布了一项具有划时代意义的专项阻断禁令,这份公告让一向倚仗长臂管辖的美...
布米普特拉北京投资基金管理有限... 美国圣路易斯联邦储备银行总裁穆萨莱姆周三发出明确信号,美联储货币政策面临的潜在风险正在发生关键转向。...
加工的秘密:超精加工设备如何做... 你知道吗? 一根头发丝的直径大约0.07毫米,也就是70微米。 超精加工设备,可切出表面,其尺寸为0...
招商证券董事长霍达因工作变动离... 北京商报讯(记者 刘宇阳 实习生 王思奕)5月8日,招商证券发布关于公司董事长离任暨推举董事代行董事...
华帝股份营收创近3年新低,37... 乐居财经李兰近日,华帝股份(002035.SZ)发布2025年年度报告。 2025年,华帝股份实现营...
大模型融资杀疯了!月之暗面狂揽... 图源:视觉中国 5月7日,据华峰资本官微消息,国内头部大模型公司月之暗面(Kimi)于近日完成新一轮...
扎根长宁二十余载,仲利国际融资... 作为总部扎根上海长宁的优质台资金融企业,仲利国际融资租赁有限公司深耕融资租赁行业二十余载,始终坚守金...
估值210亿!李彦宏又将收获一... 来源:直通IPO,文/王非 国产GPU上市潮仍然汹涌,继两家登陆A股、两家登陆H股后,这家公司正推进...
基金“盲盒”拆了 公募基金正在迎来一场让投资者“看得懂”的变革。 近日,华夏、易方达、南方、招商等12家头部及特色基金...
原创 2... 前言 十年间,中国企业在印尼镍产业链累计砸下超过140亿美元,电厂、公路、码头和全套生产线,硬生生...
原创 欧... 俄罗斯卫星通讯社5月6日报道,欧盟宣布禁止欧洲银行为含有来自不可靠供应商的关键部件的可再生能源项目提...
原创 余... 2026年5月2日,在中国理财市场扎根十三年的余额宝,终于触碰到了一个让所有人错愕的数字——7日年化...
银华基金增聘谭普景共同管理银华... 来源:新浪基金∞工作室 5月8日,银华基金管理股份有限公司发布公告称,为银华中证机器人交易型开放式指...