#1 问题描述# 2015-08-14 周五,线上乐购做活动,1元秒杀大闸蟹,库存设置为10 ,结果卖出去15个,并且线上存在脏数据,有些关联表数据不全。
执行原理:乐购调用订单平台下单,在订单平台调用促销逻辑,在促销逻辑中加限购逻辑。 #2 排查过程#
- 走查乐购秒杀的业务代码实现,在高并发的情况下,不会出现线程安全问题;【正常】
- 理解业务执行过程中,事务调用流程:Spring事务=>MyBatis事务=>druid事务(数据源)=>DB事务(数据库);【MyBatis的Batch处理导致上层Spring事务失效】
- 查看mysql binlog日志,如下:【不正常】
发现多个事务被合并成了一个事务执行,合并事务之后有啥问题?当合并事务之后,其中有一个事务执行失败后,上层的Spring就无法得知要回滚具体的哪一个事务。 4. MyBatis的配置文件中,SqlSessionTemplate配置存在问题:【不正常】
#3 解决问题#
- 注释掉SqlSessionTemplate中BATCH的配置:
#4 问题总结# Mybatis内置的ExecutorType有3种,默认的是simple,该模式下它为每个语句的执行创建一个新的预处理语句,单条提交sql;而batch模式重复使用已经预处理的语句,batch模式如果和原spring事务一起使用,将无法回滚。