MyBatis 的一级缓存导致的数据一致性问题分析

老生常谈的异常问题,这里记录一下,涉及MyBatis 的一级缓存和数据库隔离级别

目录

问题说明

下面一段示例的业务逻辑代码:

java 复制代码
@Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
@Override
public void flushOrderDetail(FlushForm form) {
    // 构建批量更新订单状态的参数集合
    List<FlushOrderParam> flushOrderParamList = payService.createFlushParam(form);
    
    // 查询出本次需要更新的订单记录
    List<OrderEntity> orderList = orderService.query(form);

    SqlSession sqlSession = null;
    try {
        // 采用MyBatis批处理模式,开启批量执行的SqlSession
        sqlSession = sqlSessionFactory.openSession(ExecutorType.BATCH);

        // 批量处理订单状态信息
        orderService.batchUpdate(sqlSession, flushOrderParamList, orderList);

        // 提交批量处理事务
        sqlSession.commit();
    } catch (Exception e) {
        // 出现异常时,回滚批处理操作
        log.error("批处理订单状态失败", e);
        if (sqlSession != null) sqlSession.rollback();

        // 将异常抛出,触发Spring事务管理的回滚机制
        throw e;
    } finally {
        // 关闭SqlSession,释放资源
        if (sqlSession != null) {
            try {
                sqlSession.close();
            } catch (Exception e) {
                log.error("关闭sqlSession异常", e);
            }
        }
    }

    // 批量操作成功后,再次查询订单,打印订单状态日志,便于排查和验证
    orderList = orderService.query(form);
    for (OrderEntity order : orderList) {
        logger.debug("订单 {} 状态 {}", order.getOrderCode(), order.getStatus());
    }
}

上面这段Spring业务代码主要内容就是想根据参数批量更新订单状态,里面使用了Spring的事务注解,同时方法中另外开启了一个数据库会话用于批处理更新,这样可以加快速度。

这段代码连接的数据库是 MySQL, 且事务隔离级别为 提交读 (READ-COMMITTED),查询MySQL事务隔离方法如下:

sql 复制代码
SELECT @@session.transaction_isolation;

问题现象:
orderList = orderService.query(form); 查询出来的订单状态没有变化,但是数据库中已经更新 !

问题原因

并非AI 给出的回答:因为两个不同的事务管理器导致不一样的结果...

首先,在提交读的隔离级别下,即便不同的事务管理器也可以相互读取到对方数据库会话 已经提交的事务数据

那为什么读取到的最后状态没有变化? 因为 MyBatis 一级缓存导致

  • MyBatis 一级缓存默认绑定在一个 SqlSession 的生命周期内下,
  • 上面代码中的开头和结尾的 orderService.query(form) 是在同一个SqlSession生命周期下
  • 且查询参数一样,这样导致两次查询结果一样,但数据库中其实已经状态更新了

问题解决

修改一级缓存的隔离级别为 ·statement· 级别,这样等同于关闭一级缓存。

yml 复制代码
mybatis:
  type-aliases-package: com.middol.*.model.**.dao
  mapper-locations: classpath:mapper/**/*Mapper.xml
  configuration:
    map-underscore-to-camel-case: true # 开启驼峰功能
    local-cache-scope: statement

主要关注 local-cache-scope: statement

或者上面代码放弃使用批处理模式,采用同一个SqlSession下操作数据库,

或者直接使用批处理的SqlSession查询订单表。

到此,问题原因和处理说明完毕!

相关推荐
2501_9418008811 小时前
Java高性能搜索引擎与Lucene实战分享:大规模文本索引、检索与优化经验
mybatis
大猫子的技术日记11 小时前
[百题重刷]前缀和 + Hash 表:缓存思想, 消除重复计算
java·缓存·哈希算法
愤怒的山羊13 小时前
jetcache List 缓存, json 序列化 泛型解析成了 JsonObject 处理
缓存·json·list
树在风中摇曳13 小时前
带哨兵位的双向循环链表详解(含 C 代码)+ LeetCode138 深度解析 + 顺序表 vs 链表缓存机制对比(图解 CPU 层级)
c语言·链表·缓存
q***428214 小时前
SpringCloud-持久层框架MyBatis Plus的使用与原理详解
spring·spring cloud·mybatis
北郭guo16 小时前
MyBatis框架讲解,工作原理、核心内容、如何实现【从浅入深】让你看完这篇文档对于MyBatis的理解更加深入
java·数据库·mybatis
斯文~16 小时前
「玩透ESA」站点配置阿里云ESA全站加速+自定义规则缓存
阿里云·缓存·云计算·cdn·esa
S***t71416 小时前
Python装饰器实现缓存
缓存
天硕国产存储技术站17 小时前
3000次零失误验证,天硕工业级SSD筑牢国产SSD安全存储方案
缓存·固态硬盘·国产ssd
前端炒粉20 小时前
35.LRU 缓存
开发语言·javascript·数据结构·算法·缓存·js