IT博客汇
  • 首页
  • 精华
  • 技术
  • 设计
  • 资讯
  • 扯淡
  • 权利声明
  • 登录 注册

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

    shendao发表于 2017-06-01 03:40:37
    love 0

    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事务一起使用,将无法回滚。



沪ICP备19023445号-2号
友情链接