Osheep

时光不回头,当下最重要。

MyBatis的Batch处理导致秒杀系统超卖的问题

1 问题描述

2015-08-14 周五,线上乐购做活动,1元秒杀大闸蟹,库存设置为10 ,结果卖出去15个,并且线上存在脏数据,有些关联表数据不全。

执行原理:乐购调用订单平台下单,在订单平台调用促销逻辑,在促销逻辑中加限购逻辑。

2 排查过程

  1. 走查乐购秒杀的业务代码实现,在高并发的情况下,不会出现线程安全问题;【正常】
  2. 理解业务执行过程中,事务调用流程:Spring事务=>MyBatis事务=>druid事务(数据源)=>DB事务(数据库);【MyBatis的Batch处理导致上层Spring事务失效】
  3. 查看mysql binlog日志,如下:【不正常】

    《MyBatis的Batch处理导致秒杀系统超卖的问题》

    查看mysql binlog日志

    发现多个事务被合并成了一个事务执行,合并事务之后有啥问题?当合并事务之后,其中有一个事务执行失败后,上层的Spring就无法得知要回滚具体的哪一个事务。

  4. MyBatis的配置文件中,SqlSessionTemplate配置存在问题:【不正常】

    《MyBatis的Batch处理导致秒杀系统超卖的问题》

    SqlSessionTemplate配置

3 解决问题

  1. 注释掉SqlSessionTemplate中BATCH的配置:

    《MyBatis的Batch处理导致秒杀系统超卖的问题》

    注释掉SqlSessionTemplate中BATCH的配置

4 问题总结

Mybatis内置的ExecutorType有3种,默认的是simple,该模式下它为每个语句的执行创建一个新的预处理语句,单条提交sql;而batch模式重复使用已经预处理的语句,batch模式如果和原spring事务一起使用,将无法回滚。

点赞