就这?Spring 事务失效场景及解决方案

小明:靓仔,我最近遇到了很邪门的事。

靓仔:哦?说来听听。

小明:上次看了你的文章《就这?一篇文章让你读懂 Spring 事务》,对事务有了详细的了解,但是在项目中还是遇到了问题,明明加了事务注解 @Transactional,却没有生效。

靓仔:那今天我就给你总结下哪些场景下事务会失效。

1、数据库引擎不支持事务

Mysql 常用的数据库引擎有 InnoDB 和 MyISAM,其中前者是支持事务的,而后者并不支持,MySQL 5.5.5 以前的默认存储引擎是:MyISAM,之前的版本默认的都是:InnoDB ,所以一定要注意自己使用的数据库支不支持事务。

2、没有被 Spring 管理

事务方法所在的类没有被注入Spring 容器,比如下面这样:

public class OrderServiceImpl implements OrderService {    
    @Autowired    
    AccountMapper accountMapper;    
    @Autowired   
    ProductMapper productMapper; 
    
    @Transactional    
    @Override    
    public void placeOrder() {        
        // 此处省略一堆逻辑                
        
        // 修用户改余额和商品库存        
        accountMapper.update();        
        productMapper.update();    
    }
}

这个类没有加 @service 注解,事务是不会生效的。

3、不是 public 方法

▲ 官方文档
▲ 翻译版本

官方文档上已经说的很清楚了,@Transactional 注解只能用于 public 方法,如果要用在非 public 方法上,可以开启 AspectJ 代理模式。

4、异常被捕获

比如下面这个例子:

@Service
public class OrderServiceImpl implements OrderService {    
    @Autowired    
    AccountMapper accountMapper;    
    @Autowired    
    ProductMapper productMapper;    
    
    @Transactional    
    @Override    
    public void placeOrder() {        
        try{            
            // 此处省略一堆逻辑                        
            
            // 修用户改余额和商品库存            
            accountMapper.update();            
            productMapper.update();        
        } catch (Exception e) {  
            
        }     
    }
}

当该方法发生异常的时候,由于异常被捕获,并没有抛出来,所以事务会失效,那这种情况下该怎么解决呢?别急,往下看

@Service
public class OrderServiceImpl implements OrderService {    
    @Autowired    
    AccountMapper accountMapper;    
    @Autowired    
    ProductMapper productMapper; 
    
    @Transactional    
    @Override    
    public void placeOrder() {       
        try{            
            // 此处省略一堆逻辑  
            
            // 修用户改余额和商品库存            
            accountMapper.update();            
            productMapper.update();        
        } catch (Exception e) {            
            // 手动回滚                             
            TransactionAspectSupport.crrentTransactionStatus().setRollbackOnly();       
        }     
    }
}

可以通过

TransactionAspectSupport.crrentTransactionStatus().setRollbackOnly();

手动进行回滚操作。

5、异常类型错误

@Transactional 注解默认只回滚 RuntimeException 类型的异常,所以在使用的时候建议修改成 Exception 类型

@Transactional(rollbackFor = Exception.class)

6、内部调用事务方法

这应该是最常见的事务失效的的场景了吧,也是我要重点讲的情况。

有些业务逻辑比较复杂的操作,比如前面例子中的下单方法,往往在写操作之前会有一堆逻辑,如果所有操作都放在一个方法里,并且加上事务,那么很可能会因为事务执行时间过长,导致事务超时,就算没超时也会影响下单接口的性能。这时可以将写操作提取出来,只对写操作加上事务,那么压力就会小很多。

请看下面这个例子:

@Service
public class OrderServiceImpl implements OrderService {    
    @Autowired    
    AccountMapper accountMapper;    
    @Autowired    
    ProductMapper productMapper;    
    
    @Override    
    public void placeOrder() {        
        // 此处省略一堆逻辑                
        this.updateByTransactional();    
    }        
    
    @Transactional    
    public void updateByTransactional() {        
        // 修用户改余额和商品库存        
        accountMapper.update();        
        productMapper.update();    
    }
}

由于发生了内部调用,而没有经过 Spring 的代理,事务就不会生效,官方文档中也有说明:

▲ 官方文档
▲ 翻译版本

那这种情况下该怎么办呢?

方案一:改为外部调用

内部调用不行,那我改成外部调用不就行了么

@Service
public class OrderServiceImpl implements OrderService {
    @Autowired
    OrderTransactionService orderTransactionService;
    
    @Override
    public void placeOrder() {
        // 此处省略一堆逻辑
        
        orderTransactionService.updateByTransactional();
    }
}

@Service
public class OrderTransactionService {
    @Autowired
    AccountMapper accountMapper;
    @Autowired
    ProductMapper productMapper;
    
    @Transactional
    public void updateByTransactional() {
        // 修用户改余额和商品库存
        accountMapper.update();
        productMapper.update();
    }
}

这是比较容易理解的一种方法

方案二:使用编程式事务

既然声明式事务有问题,那我换成编程式事务可还行?

@Service
public class OrderServiceImpl implements OrderService {
    @Autowired
    AccountMapper accountMapper;
    @Autowired
    ProductMapper productMapper;
    @Autowired
    TransactionTemplate transactionTemplate;

    @Override
    public void placeOrder() {
        // 此处省略一堆逻辑
        
        // TransactionCallbackWithoutResult 无返回参数
        // TransactionCallback 有返回参数
        transactionTemplate.execute(new TransactionCallbackWithoutResult() {
            @Override
            protected void doInTransactionWithoutResult(TransactionStatus transactionStatus) {
                try {
                    this.updateByTransactional();
                } catch (Exception e) {
                    log.error("下单失败", e);
                    transactionStatus.setRollbackOnly();
                }
            }
        });
    }
    
    public void updateByTransactional() {
        // 修用户改余额和商品库存
        accountMapper.update();
        productMapper.update();
    }
}

甭管他黑猫白猫,能抓住老鼠的就是好猫

方案三:通过外部方法调回来

这个是我看到网友提供的一种方法,又想用注解,又想自调用,那么可以参考编程式事务的方式来实现。

@Component
public class TransactionComponent {
    public interface Callback<T>{
        T run() throws Exception;
    }

    public interface CallbackWithOutResult {
        void run() throws Exception;
    }

    // 带返回参数
    @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
    @Nullable
    public <T> T doTransactional(Callback<T> callback) throws Exception {
        return callback.run();
    }

    // 无返回参数
    @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
    @Nullable
    public void doTransactionalWithOutResult(CallbackWithOutResult callbackWithOutResult) throws Exception {
        callbackWithOutResult.run();
    }
}

这样通过 TransactionComponent 调用内部方法,就可以解决失效问题了。

@Service
public class OrderServiceImpl implements OrderService {
    @Autowired
    AccountMapper accountMapper;
    @Autowired
    ProductMapper productMapper;
    @Autowired
    TransactionComponent transactionComponent;

    @Override
    public void placeOrder() {
        // 此处省略一堆逻辑
        
        transactionComponent.doTransactionalWithOutResult(() -> this.updateByTransactional());
    }
    
    public void updateByTransactional() {
        // 修用户改余额和商品库存
        accountMapper.update();
        productMapper.update();
    }
}

总结

本文总结了比较常见的几种事务失效的场景,以及一些解决方案,不一定很全。你还遇到了哪些我没提到的场景,欢迎分享,有不足之处,也欢迎指正。

END

往期推荐

就这?一篇文章让你读懂 Spring 事务

SpringBoot+Redis 实现消息订阅发布

最详细的图文解析Java各种锁(终极篇)

常见代码重构技巧,你一定用得上

图文详解 23 种设计模式

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

推荐阅读更多精彩内容