Springboot 跟着我了解下 事务 @Transactional 默认方式 Propagation.REQUIRED

 

在平常的项目里面,最常看到的就是使用注解  @Transactional   去操作事务。

 

如果稍微对spring事务有过了解的,会知道关于事务传播机制,存在7种,

也就是:

同样可以在注解@Transactional里面看到, 默认配置了的是  Propagation.REQUIRED (文章的主角)

 

 

 

为什么该篇文章,7种,我只介绍一种呢?而且是介绍的是 PROPAGATION_REQUIRED 呢?
 

因为从上文大家已经知道,默认啥都不指定的时候,我们使用的就是PROPAGATION_REQUIRED这种方式。

往往很多小伙伴在使用声明式事物的时候,就顶多加上一个异常指定,都是使用的默认传播机制变量,所以如果连这个方式都不了解的话,那也太不应该了。

我们日常操作里,对于单个方法使用事物,经常是这样:

	
    @Transactional(rollbackFor = Exception.class)
    public Boolean add(UserInfo userInfo) {
        
        //... 业务处理
        //... 业务处理
        //... 业务处理

       //手动抛异常 触发回滚等
        retrun xxx;
    }

或者说配合手动回滚使用,是这样:

	
    @Transactional(rollbackFor = Exception.class)
    public Boolean add(UserInfo userInfo) {
        
       try {
          
          //....业务逻辑处理
          
          if(XXXX){
                TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
                return false;
          }
          //....业务逻辑处理
          
          if(xxxxx){
                 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
                 return false;
          }
          

        } catch (Exception e) {
            TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
            return false;
        }
    }

以上都是单个事物方法,理解起来很简单,相信大多数场景大家就这么用一下就没有过多去理会了。

 

那么接下来就是关于 这种默认的事物传播机制  PROPAGATION_REQUIRED 我们需要关心的东西了。

 

在结合代码介绍前,先把结论贴出来: 

1.如果外部方法没有开启事务的话,Propagation.REQUIRED(默认就是)修饰的内部方法会新开启自己的事务,且开启的事务相互独立,互不干扰。


2.如果外部方法开启事务并且指定为Propagation.REQUIRED(默认就是),所有Propagation.REQUIRED修饰的内部方法和外部方法均属于同一事务 ,只要一个方法回滚,整个事务均回滚,因为大家都加入到了外部设置的这个事务里了。

 

第一种情形:

第一个业务类里面的方法,使用了声明式事务:

class testOne  
{
    @Transactional(rollbackFor = Exception.class)
    public Boolean addOne(UserInfo userInfo) {
        
        //... 业务处理
        //... 业务处理
        //... 业务处理
        retrun xxx;
    }
    
}

第二个业务类里面的方法,也使用了声明式事务:

class testTwo  
{
    @Transactional(rollbackFor = Exception.class)
    public Boolean addTwo(UserInfo userInfo) {
        
        //... 业务处理
        //... 业务处理
        //... 业务处理
        retrun xxx;
    }
    
}

然后第三个业务类里面的方法没有使用声明式事务,去调用第一个和第二个,如:

class testThree  
{
    
    public Boolean testThree(UserInfo userInfo) {
        
        addOne(xxxx);
        addTwo(xxxx);
        retrun xxx;
    }
    
}

这种情况下,addOne和addTwo两个方法的事务都是各自独立的, 也就是说,就算addOne成功了,在执行addTwo的时候出现了异常进行了回滚,并不会影响到addOne的数据。

 

 

那么,在默认addOne和addTwo都使用了事务,而且都是默认指定的传播机制PROPAGATION_REQUIRED的时候,我们想达到,只要这两个方法,或者testThree 方法,其中一个出现异常触发回滚,都可以将这三个方法一起进行回滚。那应该怎么做?

第二种情形:

让它们都加入到一个事务里面:

class testThree  
{
    @Transactional(rollbackFor = Exception.class)
    public Boolean testThree(UserInfo userInfo) {
        
        addOne(xxxx);
        addTwo(xxxx);
        retrun xxx;
    }
    
}

在testThree方法(对于addOne 和 addTwo 来说是个外部方法)上同样使用声明式事物,且也是默认指定传播机制PROPAGATION_REQUIRED。 

这样addOne事物开启时,发现外部存在指定传播机制PROPAGATION_REQUIRED的事物,那么就会加入该事物;

同样addTwo同理。

 

 

ok,简单的介绍就到此(在看完这篇,是否对事务传播的设置有了兴趣? 有的话就开始手动去做点测试,都去试试各自传播机制的情形,这样才能更灵活的去运用起来)。 

 

  • 13
    点赞
  • 37
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
@TransactionalSpring框架中用于配置事务的注解之一,它可以被应用于类或方法上。当被应用于类上时,该类中的所有公共方法都将被配置为支持事务。当被应用于方法上时,该方法将被配置为支持事务。@Transactional注解提供了多个属性,可以用于配置事务的各个方面,例如隔离级别、传播行为、只读状态等。以下是一个使用@Transactional注解配置事务的例子: ```java @Service public class UserServiceImpl implements UserService { @Autowired private UserDao userDao; @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false) public void addUser(User user) { userDao.addUser(user); } @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false) public void updateUser(User user) { userDao.updateUser(user); } @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false) public void deleteUser(int userId) { userDao.deleteUser(userId); } @Transactional(propagation = Propagation.SUPPORTS, isolation = Isolation.DEFAULT, readOnly = true) public User getUserById(int userId) { return userDao.getUserById(userId); } @Transactional(propagation = Propagation.SUPPORTS, isolation = Isolation.DEFAULT, readOnly = true) public List<User> getAllUsers() { return userDao.getAllUsers(); } } ``` 在上面的例子中,@Transactional注解被应用于UserServiceImpl类中的所有公共方法上。其中,propagation属性指定了事务的传播行为,isolation属性指定了事务的隔离级别,readOnly属性指定了事务是否只读。在这个例子中,事务的传播行为被配置为REQUIRED,表示如果当前没有事务,则创建一个新的事务;如果当前已经存在事务,则加入该事务事务的隔离级别被配置为DEFAULT,表示使用数据库默认的隔离级别。事务的只读状态被配置为false,表示该事务可以读取和修改数据。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小目标青年

对你有帮助的话,谢谢你的打赏。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值