news 2026/4/23 4:09:44

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
在一个事务里面死循环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 内有效,其他线程的操作不会使其失效。如果需要看到最新数据,需要主动绕过缓存机制。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/17 3:41:37

基于微信小程序的驾校预约与学习系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了多年的设计程序开发&#xff0c;开发过上千套设计程序&#xff0c;没有什么华丽的语言&#xff0c;只有实…

作者头像 李华
网站建设 2026/4/15 9:26:07

Flutter 本地存储方案:SharedPreferences、SQFlite 与 Hive

Flutter 本地存储方案&#xff1a;SharedPreferences、SQFlite 与 Hive 在 Flutter 应用开发中&#xff0c;本地存储是实现数据持久化的核心需求&#xff0c;广泛应用于保存用户配置、缓存网络数据、存储离线信息等场景。当前 Flutter 生态中&#xff0c;SharedPreferences、S…

作者头像 李华
网站建设 2026/4/15 1:16:07

RLHF微调场景

&#x1f34b;&#x1f34b;AI学习&#x1f34b;&#x1f34b;&#x1f525;系列专栏&#xff1a; &#x1f451;哲学语录: 用力所能及&#xff0c;改变世界。 &#x1f496;如果觉得博主的文章还不错的话&#xff0c;请点赞&#x1f44d;收藏⭐️留言&#x1f4dd;支持一下博主…

作者头像 李华
网站建设 2026/4/22 10:14:23

ThinkPad终极散热指南:双风扇智能控制完全教程

ThinkPad终极散热指南&#xff1a;双风扇智能控制完全教程 【免费下载链接】TPFanCtrl2 ThinkPad Fan Control 2 (Dual Fan) for Windows 10 and 11 项目地址: https://gitcode.com/gh_mirrors/tp/TPFanCtrl2 还在为ThinkPad风扇噪音而烦恼吗&#xff1f;TPFanCtrl2是一…

作者头像 李华
网站建设 2026/4/18 4:14:55

我发现图神经网络补全罕见病知识图谱基层漏诊率骤降

&#x1f4dd; 博客主页&#xff1a;Jax的CSDN主页 目录医生打字慢到怀疑人生&#xff1f;AI医生竟成“键盘侠”救星 一、当AI医生遇上人类医生&#xff1a;一场效率革命 二、AI医生的"作弊"秘籍&#xff1a;从打字小能手到诊断大师 三、AI医生的成长烦恼&#xff1a…

作者头像 李华
网站建设 2026/4/18 13:55:56

基于Stanley算法的自动驾驶车辆路径跟踪控制研究

摘要&#xff1a;随着自动驾驶技术的快速发展&#xff0c;车辆路径跟踪控制已成为自动驾驶系统中的关键研究内容之一。针对自动驾驶车辆在已知参考轨迹条件下的路径跟踪问题&#xff0c;本文基于车辆运动学自行车模型&#xff0c;研究并实现了一种基于 Stanley 算法 的车辆路径…

作者头像 李华