Spring事务@Transactional注解rollbackFor属性注意事项

本文详细解析了Spring框架中的@Transactional注解在处理Error和不同类型的Exception时的回滚策略,包括运行时异常、非运行时异常以及如何通过rollbackFor属性定制回滚规则,提供实验案例和总结建议。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

先回忆下java的异常模型,Throwable是最顶层的父类,有Error和Exception两个子类:

  1. Error表示严重的错误(如OOM等);
  2. Exception可以分为运行时异常(RuntimeException及其子类)和非运行时异常(Exception的子类中,除了RuntimeException及其子类之外的类)。

        非运行时异常是检查异常(checked exceptions),如IOException、SQLException等以及用户自定义的Exception异常。对于这种异常,JAVA编译器强制要求我们必需对出现的这些异常进行try catch并处理,否则程序就不能编译通过。因为这类异常是可预料的,编译阶段就检查的出来;
        Error和运行时异常是非检查异常(unchecked exceptions),不需要try catch,因为这类异常是不可预料的,编辑阶段不会检查,没必要检查,也检查不出来。


 查看源码

         spring的@Transactional注解可以很方便的开启事务,但是默认只在遇到运行时异常和Error时才会回滚,非运行时异常不回滚,即Exception的子类中,除了RuntimeException及其子类,其他的类默认不回滚(不知道为什么要这样设计?)
而rollbackFor属性可以解决这个问题,rollbackFor = Exception.class表示Exception及其子类的异常都会触发回滚,同时不影响Error的回滚。


 测试

下面是关于@Transactional注解的一些实验

实验一

不加rollbackFor属性,抛出RuntimeException,正常回滚

    @Transactional
    public void save(){
        StudentDO studentDO = new StudentDO();
        studentDO.setName("ltm");
        studentDO.setAge(22);
        studentMapper.insert(studentDO);

        throw new RuntimeException("我是异常");
    }

 实验二

不加rollbackFor属性,抛出IOException,不回滚

    @Transactional
    public void save() throws IOException{
        StudentDO studentDO = new StudentDO();
        studentDO.setName("ltm");
        studentDO.setAge(22);
        studentMapper.insert(studentDO);

        throw new IOException();
    }

实验三

加上rollbackFor = Exception.class,抛出IOException,正常回滚

    @Transactional(rollbackFor = Exception.class)
    public void save() throws IOException{
        StudentDO studentDO = new StudentDO();
        studentDO.setName("ltm");
        studentDO.setAge(22);
        studentMapper.insert(studentDO);

        throw new IOException();
    }

实验四

不加rollbackFor属性,抛出OutOfMemoryError,正常回滚

    @Transactional()
    public void save(){
        StudentDO studentDO = new StudentDO();
        studentDO.setName("ltm");
        studentDO.setAge(22);
        studentMapper.insert(studentDO);

        throw new OutOfMemoryError();
    }

实验五

加上rollbackFor = Exception.class,抛出OutOfMemoryError,正常回滚,说明rollbackFor = Exception.class不会覆盖Error的回滚

    @Transactional(rollbackFor = Exception.class)
    public void save(){
        StudentDO studentDO = new StudentDO();
        studentDO.setName("ltm");
        studentDO.setAge(22);
        studentMapper.insert(studentDO);

        throw new OutOfMemoryError();
    }

总结

        @Transactional只能回滚RuntimeException和RuntimeException下面的子类抛出的异常 不能回滚Exception异常。如果需要支持回滚Exception异常请用@Transactional(rollbackFor = Exception.class)

        增删改的时候建议使用@Transactional(rollbackFor = Exception.class)

@Transactional一些失效的场景:

  1. 不是用public修饰
  2. try catch捕获了异常(没有在catch里面手动抛出异常)
  3. 没有加@Service(也就是没有被 Spring 管理)

@Transactional的相关属性

属性类型描述
valueString可选的限定描述符,指定使用的事务管理器
propagationenum: Propagation可选的事务传播行为设置
isolationenum: Isolation可选的事务隔离级别设置
readOnlyboolean读写或只读事务,默认读写
timeoutint (in seconds granularity)事务超时时间设置
rollbackForClass对象数组,必须继承自Throwable导致事务回滚的异常类数组
rollbackForClassName类名数组,必须继承自Throwable导致事务回滚的异常类名字数组
noRollbackForClass对象数组,必须继承自Throwable不会导致事务回滚的异常类数组
noRollbackForClassName类名数组,必须继承自Throwable不会导致事务回滚的异常类名字数组

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值