深入理解Spring的事务传播行为

前言

Spring在TransactionDefinition接口中规定了7种类型的事务传播行为。事务传播行为是Spring框架独有的事务增强特性,他不属于的事务实际提供方数据库行为。这是Spring为我们提供的强大的工具箱,使用事务传播行可以为我们的开发工作提供许多便利。但是人们对他的误解也颇多,你一定也听过“service方法事务最好不要嵌套”的传言。要想正确的使用工具首先需要了解工具。本文对七种事务传播行为做详细介绍,内容主要代码示例的方式呈现。

基础概念

1. 什么是事务传播行为?

事务传播行为用来描述由某一个事务传播行为修饰的方法被嵌套进另一个方法的时事务如何传播。

用伪代码说明:

 public void methodA(){
 methodB();
 //doSomething
 }

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

代码中methodA()方法嵌套调用了methodB()方法,methodB()的事务传播行为由@Transaction(Propagation=XXX)设置决定。这里需要注意的是methodA()并没有开启事务,某一个事务传播行为修饰的方法并不是必须要在开启事务的外围方法中调用。

2. Spring中七种事务传播行为

事务传播行为类型 说明
PROPAGATION_REQUIRED 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。这是最常见的选择。
PROPAGATION_SUPPORTS 支持当前事务,如果当前没有事务,就以非事务方式执行。
PROPAGATION_MANDATORY 使用当前的事务,如果当前没有事务,就抛出异常。
PROPAGATION_REQUIRES_NEW 新建事务,如果当前存在事务,把当前事务挂起。
PROPAGATION_NOT_SUPPORTED 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
PROPAGATION_NEVER 以非事务方式执行,如果当前存在事务,则抛出异常。
PROPAGATION_NESTED 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。

定义非常简单,也很好理解,下面我们就进入代码测试部分,验证我们的理解是否正确。

代码验证

文中代码以传统三层结构中两层呈现,即Service和Dao层,由Spring负责依赖注入和注解式事务管理,DAO层由Mybatis实现,你也可以使用任何喜欢的方式,例如,Hibernate,JPA,JDBCTemplate等。数据库使用的是MySQL数据库,你也可以使用任何支持事务的数据库,并不会影响验证结果。

首先我们在数据库中创建两张表:

user1

CREATE TABLE `user1` (
 `id` INTEGER UNSIGNED NOT NULL AUTO_INCREMENT,
 `name` VARCHAR(45) NOT NULL DEFAULT '',
 PRIMARY KEY(`id`)
)
ENGINE = InnoDB;

user2

CREATE TABLE `user2` (
 `id` INTEGER UNSIGNED NOT NULL AUTO_INCREMENT,
 `name` VARCHAR(45) NOT NULL DEFAULT '',
 PRIMARY KEY(`id`)
)
ENGINE = InnoDB;

然后编写相应的Bean和DAO层代码:

User1

public class User1 {
 private Integer id;
 private String name;
 //get和set方法省略...
}

User2

public class User2 {
 private Integer id;
 private String name;
 //get和set方法省略...
}

User1Mapper

public interface User1Mapper {
 int insert(User1 record);
 User1 selectByPrimaryKey(Integer id);
 //其他方法省略...
}

User2Mapper

public interface User2Mapper {
 int insert(User2 record);
 User2 selectByPrimaryKey(Integer id);
 //其他方法省略...
}

最后也是具体验证的代码由service层实现,下面我们分情况列举。

1.PROPAGATION_REQUIRED

我们为User1Service和User2Service相应方法加上Propagation.REQUIRED属性。

User1Service方法:

@Service
public class User1ServiceImpl implements User1Service {
 //省略其他...
 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void addRequired(User1 user){
 user1Mapper.insert(user);
 }
}

User2Service方法:

@Service
public class User2ServiceImpl implements User2Service {
 //省略其他...
 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void addRequired(User2 user){
 user2Mapper.insert(user);
 }
 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void addRequiredException(User2 user){
 user2Mapper.insert(user);
 throw new RuntimeException();
 }
}

1.1 场景一

此场景外围方法没有开启事务。

验证方法1:

 @Override
 public void notransaction_exception_required_required(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequired(user2);
 throw new RuntimeException();
 }

验证方法2:

 @Override
 public void notransaction_required_required_exception(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiredException(user2);
 }

分别执行验证方法,结果:

验证方法序号 数据库结果 结果分析

验证方法序号 数据库结果 结果分析
1 “张三”、“李四”均插入。 外围方法未开启事务,插入“张三”、“李四”方法在自己的事务中独立运行,外围方法异常不影响内部插入“张三”、“李四”方法独立的事务。
2 “张三”插入,“李四”未插入。 外围方法没有事务,插入“张三”、“李四”方法都在自己的事务中独立运行,所以插入“李四”方法抛出异常只会回滚插入“李四”方法,插入“张三”方法不受影响。

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

1.2 场景二

外围方法开启事务,这个是使用率比较高的场景。

验证方法1:

 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void transaction_exception_required_required(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequired(user2);
 throw new RuntimeException();
 }

验证方法2:

 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void transaction_required_required_exception(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiredException(user2);
 }

验证方法3:

 @Transactional
 @Override
 public void transaction_required_required_exception_try(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 try {
 user2Service.addRequiredException(user2);
 } catch (Exception e) {
 System.out.println("方法回滚");
 }
 }

分别执行验证方法,结果:

验证方法序号 数据库结果 结果分析
1 “张三”、“李四”均未插入。 外围方法开启事务,内部方法加入外围方法事务,外围方法回滚,内部方法也要回滚。
2 “张三”、“李四”均未插入。 外围方法开启事务,内部方法加入外围方法事务,内部方法抛出异常回滚,外围方法感知异常致使整体事务回滚。
3 “张三”、“李四”均未插入。 外围方法开启事务,内部方法加入外围方法事务,内部方法抛出异常回滚,即使方法被catch不被外围方法感知,整个事务依然回滚。

结论:以上试验结果我们证明在外围方法开启事务的情况下Propagation.REQUIRED修饰的内部方法会加入到外围方法的事务中,所有Propagation.REQUIRED修饰的内部方法和外围方法均属于同一事务,只要一个方法回滚,整个事务均回滚。

2.PROPAGATION_REQUIRES_NEW

我们为User1Service和User2Service相应方法加上Propagation.REQUIRES_NEW属性。

User1Service方法:

@Service
public class User1ServiceImpl implements User1Service {
 //省略其他...
 @Override
 @Transactional(propagation = Propagation.REQUIRES_NEW)
 public void addRequiresNew(User1 user){
 user1Mapper.insert(user);
 }
 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void addRequired(User1 user){
 user1Mapper.insert(user);
 }
}

User2Service方法:

@Service
public class User2ServiceImpl implements User2Service {
 //省略其他...
 @Override
 @Transactional(propagation = Propagation.REQUIRES_NEW)
 public void addRequiresNew(User2 user){
 user2Mapper.insert(user);
 }
 @Override
 @Transactional(propagation = Propagation.REQUIRES_NEW)
 public void addRequiresNewException(User2 user){
 user2Mapper.insert(user);
 throw new RuntimeException();
 }
}

2.1 场景一

外围方法没有开启事务。

验证方法1:

 @Override
 public void notransaction_exception_requiresNew_requiresNew(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequiresNew(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiresNew(user2);
 throw new RuntimeException();
 }

验证方法2:

 @Override
 public void notransaction_requiresNew_requiresNew_exception(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequiresNew(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiresNewException(user2);
 }

分别执行验证方法,结果:

验证方法序号 数据库结果 结果分析
1 “张三”插入,“李四”插入。 外围方法没有事务,插入“张三”、“李四”方法都在自己的事务中独立运行,外围方法抛出异常回滚不会影响内部方法。
2 “张三”插入,“李四”未插入 外围方法没有开启事务,插入“张三”方法和插入“李四”方法分别开启自己的事务,插入“李四”方法抛出异常回滚,其他事务不受影响。

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

2.2 场景二

外围方法开启事务。

验证方法1:

 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void transaction_exception_required_requiresNew_requiresNew(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiresNew(user2);
 User2 user3=new User2();
 user3.setName("王五");
 user2Service.addRequiresNew(user3);
 throw new RuntimeException();
 }

验证方法2:

 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void transaction_required_requiresNew_requiresNew_exception(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiresNew(user2);
 User2 user3=new User2();
 user3.setName("王五");
 user2Service.addRequiresNewException(user3);
 }

验证方法3:

 @Override
 @Transactional(propagation = Propagation.REQUIRED)
 public void transaction_required_requiresNew_requiresNew_exception_try(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addRequired(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addRequiresNew(user2);
 User2 user3=new User2();
 user3.setName("王五");
 try {
  user2Service.addRequiresNewException(user3);
 } catch (Exception e) {
  System.out.println("回滚");
 }
 }

分别执行验证方法,结果:

验证方法序号 数据库结果 结果分析
1 “张三”未插入,“李四”插入,“王五”插入。 外围方法开启事务,插入“张三”方法和外围方法一个事务,插入“李四”方法、插入“王五”方法分别在独立的新建事务中,外围方法抛出异常只回滚和外围方法同一事务的方法,故插入“张三”的方法回滚。
2 “张三”未插入,“李四”插入,“王五”未插入。 外围方法开启事务,插入“张三”方法和外围方法一个事务,插入“李四”方法、插入“王五”方法分别在独立的新建事务中。插入“王五”方法抛出异常,首先插入 “王五”方法的事务被回滚,异常继续抛出被外围方法感知,外围方法事务亦被回滚,故插入“张三”方法也被回滚。
3 “张三”插入,“李四”插入,“王五”未插入。 外围方法开启事务,插入“张三”方法和外围方法一个事务,插入“李四”方法、插入“王五”方法分别在独立的新建事务中。插入“王五”方法抛出异常,首先插入“王五”方法的事务被回滚,异常被catch不会被外围方法感知,外围方法事务不回滚,故插入“张三”方法插入成功。

结论:在外围方法开启事务的情况下Propagation.REQUIRES_NEW修饰的内部方法依然会单独开启独立事务,且与外部方法事务也独立,内部方法之间、内部方法和外部方法事务均相互独立,互不干扰。

3.PROPAGATION_NESTED

我们为User1Service和User2Service相应方法加上Propagation.NESTED属性。

User1Service方法:

@Service
public class User1ServiceImpl implements User1Service {
 //省略其他...
 @Override
 @Transactional(propagation = Propagation.NESTED)
 public void addNested(User1 user){
 user1Mapper.insert(user);
 }
}

User2Service方法:

@Service
public class User2ServiceImpl implements User2Service {
 //省略其他...
 @Override
 @Transactional(propagation = Propagation.NESTED)
 public void addNested(User2 user){
 user2Mapper.insert(user);
 }

 @Override
 @Transactional(propagation = Propagation.NESTED)
 public void addNestedException(User2 user){
 user2Mapper.insert(user);
 throw new RuntimeException();
 }
}

3.1 场景一

此场景外围方法没有开启事务。

验证方法1:

 @Override
 public void notransaction_exception_nested_nested(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addNested(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addNested(user2);
 throw new RuntimeException();
 }

验证方法2:

 @Override
 public void notransaction_nested_nested_exception(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addNested(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addNestedException(user2);
 }

分别执行验证方法,结果:

验证方法序号 数据库结果 结果分析
1 “张三”、“李四”均插入。 外围方法未开启事务,插入“张三”、“李四”方法在自己的事务中独立运行,外围方法异常不影响内部插入“张三”、“李四”方法独立的事务。
2 “张三”插入,“李四”未插入。 外围方法没有事务,插入“张三”、“李四”方法都在自己的事务中独立运行,所以插入“李四”方法抛出异常只会回滚插入“李四”方法,插入“张三”方法不受影响。

结论:通过这两个方法我们证明了在外围方法未开启事务的情况下Propagation.NESTEDPropagation.REQUIRED作用相同,修饰的内部方法都会新开启自己的事务,且开启的事务相互独立,互不干扰。

3.2 场景二

外围方法开启事务。

验证方法1:

 @Transactional
 @Override
 public void transaction_exception_nested_nested(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addNested(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addNested(user2);
 throw new RuntimeException();
 }

验证方法2:

 @Transactional
 @Override
 public void transaction_nested_nested_exception(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addNested(user1);
 User2 user2=new User2();
 user2.setName("李四");
 user2Service.addNestedException(user2);
 }

验证方法3:

 @Transactional
 @Override
 public void transaction_nested_nested_exception_try(){
 User1 user1=new User1();
 user1.setName("张三");
 user1Service.addNested(user1);
 User2 user2=new User2();
 user2.setName("李四");
 try {
  user2Service.addNestedException(user2);
 } catch (Exception e) {
  System.out.println("方法回滚");
 }
 }

分别执行验证方法,结果:

验证方法序号 数据库结果 结果分析
1 “张三”、“李四”均未插入。 外围方法开启事务,内部事务为外围事务的子事务,外围方法回滚,内部方法也要回滚。
2 “张三”、“李四”均未插入。 外围方法开启事务,内部事务为外围事务的子事务,内部方法抛出异常回滚,且外围方法感知异常致使整体事务回滚。
3 “张三”插入、“李四”未插入。 外围方法开启事务,内部事务为外围事务的子事务,插入“张三”内部方法抛出异常,可以单独对子事务回滚。

结论:以上试验结果我们证明在外围方法开启事务的情况下Propagation.NESTED修饰的内部方法属于外部事务的子事务,外围主事务回滚,子事务一定回滚,而内部子事务可以单独回滚而不影响外围主事务和其他子事务

4. REQUIRED,REQUIRES_NEW,NESTED异同

由“1.2 场景二”和“3.2 场景二”对比,我们可知:

NESTED和REQUIRED修饰的内部方法都属于外围方法事务,如果外围方法抛出异常,这两种方法的事务都会被回滚。但是REQUIRED是加入外围方法事务,所以和外围事务同属于一个事务,一旦REQUIRED事务抛出异常被回滚,外围方法事务也将被回滚。而NESTED是外围方法的子事务,有单独的保存点,所以NESTED方法抛出异常被回滚,不会影响到外围方法的事务。

由“2.2 场景二”和“3.2 场景二”对比,我们可知:

NESTED和REQUIRES_NEW都可以做到内部方法事务回滚而不影响外围方法事务。但是因为NESTED是嵌套事务,所以外围方法回滚之后,作为外围方法事务的子事务也会被回滚。而REQUIRES_NEW是通过开启新的事务实现的,内部事务和外围事务是两个事务,外围事务回滚不会影响内部事务。

5. 其他事务传播行为

鉴于文章篇幅问题,其他事务传播行为的测试就不在此一一描述了,感兴趣的读者可以去源码中自己寻找相应测试代码和结果解释。传送门:https://github.com/TmTse/tran...

模拟用例

介绍了这么多事务传播行为,我们在实际工作中如何应用呢?下面我来举一个示例:

假设我们有一个注册的方法,方法中调用添加积分的方法,如果我们希望添加积分不会影响注册流程(即添加积分执行失败回滚不能使注册方法也回滚),我们会这样写:

 @Service
 public class UserServiceImpl implements UserService {
 @Transactional
 public void register(User user){
  try {
  membershipPointService.addPoint(Point point);
  } catch (Exception e) {
  //省略...
  }
  //省略...
 }
 //省略...
 }

我们还规定注册失败要影响addPoint()方法(注册方法回滚添加积分方法也需要回滚),那么addPoint()方法就需要这样实现:

 @Service
 public class MembershipPointServiceImpl implements MembershipPointService{
 @Transactional(propagation = Propagation.NESTED)
 public void addPoint(Point point){
  try {
  recordService.addRecord(Record record);
  } catch (Exception e) {
  //省略...
  }
  //省略...
 }
 //省略...
 }

我们注意到了在addPoint()中还调用了addRecord()方法,这个方法用来记录日志。他的实现如下:

 @Service
 public class RecordServiceImpl implements RecordService{
 @Transactional(propagation = Propagation.NOT_SUPPORTED)
 public void addRecord(Record record){
  //省略...
 }
 //省略...
 }

我们注意到addRecord()方法中propagation = Propagation.NOT_SUPPORTED ,因为对于日志无所谓精确,可以多一条也可以少一条,所以addRecord()方法本身和外围addPoint()方法抛出异常都不会使addRecord()方法回滚,并且addRecord()方法抛出异常也不会影响外围addPoint()方法的执行。

通过这个例子相信大家对事务传播行为的使用有了更加直观的认识,通过各种属性的组合确实能让我们的业务实现更加灵活多样。

结论

通过上面的介绍,相信大家对Spring事务传播行为有了更加深入的理解,希望大家日常开发工作有所帮助。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • 深入理解Spring事务的传播行为

    前言 本文主要介绍下Spring事务中的传播行为.事务传播行为是Spring框架独有的事务增强特性,他不属于的事务实际提供方数据库行为.这是Spring为我们提供的强大的工具箱,使用事务传播行可以为我们的开发工作提供许多便利. 下面话不多说了,来一起看看详细的介绍吧 事务传播行为介绍 Spring中的7个事务传播行为: |事务行为|说明 | |:--|:--| |PROPAGATION_REQUIRED | 支持当前事务,假设当前没有事务.就新建一个事务 | | PROPAGATION_SUPP

  • Spring事务传播属性和隔离级别详细介绍

    1 事务的传播属性(Propagation) 1) REQUIRED ,这个是默认的属性 Support a current transaction, create a new one if none exists. 如果存在一个事务,则支持当前事务.如果没有事务则开启一个新的事务. 被设置成这个级别时,会为每一个被调用的方法创建一个逻辑事务域.如果前面的方法已经创建了事务,那么后面的方法支持当前的事务,如果当前没有事务会重新建立事务. 2) MANDATORY Support a curren

  • Spring中事务传播行为的介绍

    传播行为定义关于客户端和被调用方法的事务边界.Spring定义了7种截然不同的传播行为. 1,PROPAGATION_MANDATORY,表示该方法必须运行在一个事务中.如果当前没有事务正在发生,则抛出异常. 2,PROPAGATION_NESTED,表示如果当前有一个事务正在运行当中,则该方法应该运行在一个嵌套事务中.被嵌套的事务可以独立于封装事务进行提交或回滚.如果封装事务不存在,则行为就像PROPAGATION_REQUIRED一样. 3,PROPAGATION_NEVER,表示当前方法不

  • Spring 事务隔离与事务传播的详解与对比

    Spring 事务隔离与事务传播的详解与对比 Spring是SSH中的管理员,负责管理其它框架,协调各个部分的工作.今天一起学习一下Spring的事务管理.Spring的事务管理分为声明式跟编程式.声明式就是在Spring的配置文件中进行相关配置:编程式就是用注解的方式写到代码里. Spring配置文件中关于事务配置总是由三个组成部分,分别是DataSource.TransactionManager和代理机制这三部分,无论哪种配置方式,一般变化的只是代理机制这部分. DataSource. Tr

  • 深入理解Spring的事务传播行为

    前言 Spring在TransactionDefinition接口中规定了7种类型的事务传播行为.事务传播行为是Spring框架独有的事务增强特性,他不属于的事务实际提供方数据库行为.这是Spring为我们提供的强大的工具箱,使用事务传播行可以为我们的开发工作提供许多便利.但是人们对他的误解也颇多,你一定也听过"service方法事务最好不要嵌套"的传言.要想正确的使用工具首先需要了解工具.本文对七种事务传播行为做详细介绍,内容主要代码示例的方式呈现. 基础概念 1. 什么是事务传播行

  • 老生常谈spring的事务传播机制

    目录 spring的事务传播机制 1.why 为什么会有事务传播机制? 2.传播机制生效的条件 解决方案 3.传播机制类型 PROPAGATION_REQUIRED (默认) REQUIRES_NEW (一般用在子方法需要单独事务) NESTED SUPPORTS NOT_SUPPORTED MANDATORY NEVER spring的事务传播机制 背景:实习期间几次遇到事务方法,有一次本地测试时发现事务没有回滚,就把简单描述写在wx上,今天来给spring事务做个自我总结. 1.why 为什

  • 深入了解Spring的事务传播机制

    目录 Spring 事务传播机制有哪些? 事务传播机制使用与演示 REQUIRED 使用演示 REQUIRED_NEW 使用演示 NESTED 使用演示 总结 Spring 事务传播机制是指,包含多个事务的方法在相互调用时,事务是如何在这些方法间传播的. 既然是“事务传播”,所以事务的数量应该在两个或两个以上,Spring 事务传播机制的诞生是为了规定多个事务在传播过程中的行为的.比如方法 A 开启了事务,而在执行过程中又调用了开启事务的 B 方法,那么 B 方法的事务是应该加入到 A 事务当中

  • 深入理解Spring事务原理

    一.事务的基本原理 Spring事务的本质其实就是数据库对事务的支持,没有数据库的事务支持,spring是无法提供事务功能的.对于纯JDBC操作数据库,想要用到事务,可以按照以下步骤进行:  1.获取连接 Connection con = DriverManager.getConnection()  2.开启事务con.setAutoCommit(true/false);  3.执行CRUD  4.提交事务/回滚事务 con.commit() / con.rollback();  5.关闭连接

  • Spring事务传播中嵌套调用实现方法详细介绍

    目录 前言 7种传播方式 注解式事务 事务的方法之间的调用 注意事项 前言 最近在使用Spring框架时遇到了一些问题,主要是Spring的事务传播问题,一个不带事务的方法调用带事务的方法,有时候会出现不回滚的情况,所以写了这篇文章来记录一下. 7种传播方式 我们先来看Spring事务的7中传播方式以及对应的描述 属性名称 值 描述 PROPAGATION__REQUIRED REQUIRED 表示的是当前这个方法必须运行在一个事务环境中,如果当前方法已经处于事务环境中,就可以直接使用该方法,否

  • 关于spring事务传播行为非事务方式的理解

    目录 spring事务传播行为非事务方式 Spring事务理解和配置 1 Spring事务 1.1 事务简介 1.2 事务的传播行为 1.3 编程式事务(需要手动写代码,了解) 1.4 声明式事务(xml配置) 1.5 声明式事务(注解) spring事务传播行为非事务方式 具体的隔离级别可以看看搜一下,相关的博客挺多的,现在说一下对传播行为 PROPAGATION_NOT_SUPPORTED.PROPAGATION_NEVER中很多解释为 非事务方式运行 的理解.    关于什么是'非事务方式

随机推荐