当前位置: 首页 > news >正文

在一个事务里面死循环select一条数据,当我开启事务时,数据是1,每过5秒我就select一次,这个时候mybatis的一级缓存起作用了,所以不会去数据库查数据,等别的线程更新了数据表的数据,会使m

这个问题涉及到 MyBatis 一级缓存、事务隔离级别和数据库的并发控制。让我详细分析一下:

1. MyBatis 一级缓存的行为

MyBatis 一级缓存(Local Cache)是SqlSession 级别​ 的缓存:

  • 在同一个 SqlSession 中执行相同的查询,MyBatis 会返回缓存中的结果

  • 默认情况下,一级缓存是开启的

2. 重要结论

不会失效,原因如下:

2.1 MyBatis 一级缓存的失效条件

MyBatis 一级缓存会在以下情况下失效:

  • 执行了 INSERT/UPDATE/DELETE 操作

  • 调用了sqlSession.clearCache()方法

  • 执行了带有flushCache="true"的查询

  • SqlSession 关闭

但是其他线程的更新不会导致当前线程的 MyBatis 一级缓存失效,因为一级缓存是 SqlSession 私有的。

3. 实际演示代码

@Service public class DemoService { @Autowired private SqlSessionFactory sqlSessionFactory; // 线程1:在事务中循环查询 public void thread1Transaction() { SqlSession sqlSession = sqlSessionFactory.openSession(TransactionIsolationLevel.READ_COMMITTED); try { for (int i = 0; i < 10; i++) { // 每次查询都会从一级缓存获取 User user = sqlSession.selectOne("com.example.mapper.UserMapper.selectById", 1L); System.out.println("Thread1 - 查询结果: " + user.getCount()); Thread.sleep(5000); // 等待5秒 } } finally { sqlSession.close(); } } // 线程2:更新数据 public void thread2Update() { SqlSession sqlSession = sqlSessionFactory.openSession(); try { sqlSession.update("com.example.mapper.UserMapper.incrementCount", 1L); sqlSession.commit(); // 提交事务 } finally { sqlSession.close(); } } }

4. 解决方案

如果你希望看到其他线程的更新,有以下几种方式:

4.1 关闭 MyBatis 一级缓存(针对特定查询)

<!-- 在mapper.xml中 --> <select id="selectById" resultType="User" flushCache="true"> SELECT * FROM user WHERE id = #{id} </select>

4.2 手动清除缓存

// 在查询前清除缓存 sqlSession.clearCache(); User user = sqlSession.selectOne("com.example.mapper.UserMapper.selectById", 1L);

4.3 使用不同的 SqlSession

// 每次查询创建新的 SqlSession for (int i = 0; i < 10; i++) { try (SqlSession newSession = sqlSessionFactory.openSession()) { User user = newSession.selectOne("com.example.mapper.UserMapper.selectById", 1L); System.out.println("查询结果: " + user.getCount()); } Thread.sleep(5000); }

4.4 配置二级缓存(不推荐用于此场景)

二级缓存是 Mapper 级别的,可以被多个 SqlSession 共享,但需要考虑同步问题。

5. 事务隔离级别的影响

即使 MyBatis 每次都去数据库查询,事务隔离级别也会影响你看到的数据:

// 设置事务隔离级别为 READ_COMMITTED @Transactional(isolation = Isolation.READ_COMMITTED) public void queryInTransaction() { for (int i = 0; i < 10; i++) { User user = userMapper.selectById(1L); System.out.println("查询结果: " + user.getCount()); Thread.sleep(5000); } }

事务隔离级别

  • REPEATABLE_READ(MySQL默认):同一事务中多次读取相同数据,结果一致

  • READ_COMMITTED:可以读取到其他事务已提交的修改

6. 推荐做法

如果需要在长事务中看到其他线程的更新:

  1. 使用 READ_COMMITTED 隔离级别

  2. 关闭 MyBatis 一级缓存​ 或每次查询前清空缓存

  3. 或者使用新的 SqlSession 查询

@Transactional(isolation = Isolation.READ_COMMITTED) public void queryWithLatestData() { for (int i = 0; i < 10; i++) { // 方案1:清空缓存 clearCacheAndQuery(); // 方案2:新建SqlSession查询 queryWithNewSession(); Thread.sleep(5000); } }

核心要点:MyBatis 一级缓存只在当前 SqlSession 内有效,其他线程的操作不会使其失效。如果需要看到最新数据,需要主动绕过缓存机制。

http://www.cnnetsun.cn/news/94319.html

相关文章:

  • 字符设备驱动(5)
  • Flutter 表单开发实战:表单验证、输入格式化与提交处理
  • 【光子 AI】AI Agent 架构师 / 技术专家 10 道必考面试题和必过答案完整讲解 1
  • Flutter 主题与深色模式:全局样式统一与动态切换
  • 基于 GEE 使用 Sentinel-2 遥感影像数据反演水体叶绿素 a 质量浓度
  • 小红书数据采集架构解析与工程实践
  • 长沙对非合作深化 探索新型易货贸易
  • OpenCore Legacy Patcher终极教程:让老旧Mac完美运行最新macOS
  • 1、开启GIMP图像编辑之旅:从安装到精通
  • 2、开启 GIMP 图形编辑之旅
  • 怎么建立一套高效的设备运维管理体系?
  • 小爱音箱AI升级:让你的智能音箱秒变高智商语音助手
  • UnrealPakViewer终极指南:从入门到精通的Pak文件分析完整教程
  • 俄罗斯T-Tech公司推出T-pro 2.0:让AI说俄语更流利混合智能模型
  • MCP智能体连接协议面临企业级挑战
  • 联想发布数据存储新品助力企业AI发展
  • 人工智能使用大揭秘:OpenRouter公司百万亿规模数据分析报告
  • 微信DAT文件转换神器,牛批了
  • OBS音频插件实用技巧:专业级直播音效快速配置指南
  • BetterNCM插件配置全攻略:5步打造你的专属音乐工作站
  • 如何5分钟掌握网盘下载加速:告别限速的终极方案
  • 26年找实习的前端,建议跟着飞书准备面试...
  • 毕业论文知网AIGC怎么降?推荐10款免费降AI工具,完美保留原格式且无AI味!
  • “草台班子”的真相:为什么说未来十年,普通人逆袭的机会在这里?
  • 知网/维普AIGC怎么降?2025年降AI率工具大盘点,AI率低于10%且完美保留原格式!
  • 2025年降AI率哪个工具靠谱?5个工具核心优势解析,笔灵降AI性价比领先!
  • 年终总结,爆梗朋友圈
  • springboot健康管理小程序
  • PDF对比终极方案:如何用diff-pdf快速找出文档差异
  • Burp Suite抓包失败的5个常见原因及解决方法,第3个最容易被忽略!