【MyBatis】RC隔离级别下,MyBatis 一级缓存、二级缓存造成的不一致情况

news2025/1/14 18:23:01

前言

数据库的MVCC 及 锁机制保证了数据的隔离、一致性。而建立在数据库之上的缓存,都会破坏掉数据库的一致性保障。本文探索在RC隔离级别下,MyBatis 一级缓存、二级缓存造成的坑。顺便复习一下 Spock 的用法,更加体验到了 Groovy 清爽的语法。

本文内容的代码仓(分支L1Cache 、L2Cache )
MyBatis 官网缓存部分

  • 体验一下 Spock 的一个测试用例 (本文与Spring Boot 进行集成)
    def "RR环境下, 缓存不影响结果"() {
        given:
        mapper.insert(1, "james")
        def name1
        def name2
        transaction.execute {
            // 当前线程查询
            name1 = mapper.selectNameByIdWithoutL1Cache(1)

            // 新开线程更新并提交
            executor.execute {
                transaction.execute {
                    mapper.updateNameById("kobe", 1)
                }
            }

            // 确保更新线程提交成功
            Thread.sleep(2000)

            // 使用不同的mapper保证不命中二级缓存
            name2 = anotherMapper.selectNameById(1)
        }
        Thread.sleep(3000)
        
        expect:
        name1 == "james"
        // RC 下这个结果未 "kobe"
        name2 == "james"
    }

缓存的结构

在这里插入图片描述

关闭二级缓存,研究一级缓存

  • 关闭全局缓存
mybatis:
  mapper-locations: mapper/*Mapper.xml
  map-underscore-to-camel-case: true
  cache-enabled: false
  • mybatis 标签的默认情况即开启了一级缓存(官网的内容:不声明的情况,标签上的两个属性如下)
<select ... flushCache="false" useCache="true"/>

值得一提的是 flushCache=“true” 的话会禁用所有缓存

  • RC 重复读无法读最新提交的数据,测试用例:
 def "证明 Mybatis 存在一级缓存, 且破坏了RC的事务隔离能力"() {
        given:
        mapper.insert(1, "james")
        def name1
        def name2
        transaction.execute {
            // 当前线程查询
            name1 = mapper.selectNameById(1)

            // 新开线程更新并提交
            executor.execute {
                transaction.execute {
                    mapper.updateNameById("kobe", 1)
                }
            }

            // 确保更新线程提交成功
            Thread.sleep(2000)

            // 当前线程命中缓存并返回, 忽略了更新的值"kobe"
            name2 = mapper.selectNameById(1)
        }
        Thread.sleep(3000)
        expect:
        name1 == "james"
        name2 == "james"
    }

关闭一级缓存、研究二级缓存

  • 关闭一级缓存
<select ... useCache="false"/>
  • 开启二级缓存
mybatis:
  mapper-locations: mapper/*Mapper.xml
  map-underscore-to-camel-case: true
  cache-enabled: true 
  • 二级缓存影响RC,测试用例:
 def "证明 Mybatis 如果配置二级缓存 (关闭一级缓存), 且破坏了RC的事务隔离能力"() {
        given:
        mapper.insert(1, "james")
        def name1
        def name2
        transaction.execute {
            // 当前线程查询
            name1 = mapper.selectNameByIdWithoutL1Cache(1)

            // 新开线程更新并提交
            executor.execute {
                transaction.execute {
                    anotherMapper.updateNameById("kobe", 1)
                }
            }

            // 确保更新线程提交成功
            Thread.sleep(2000)

            // 当前线程命中缓存并返回, 忽略了更新的值"kobe"
            name2 = mapper.selectNameByIdWithoutL1Cache(1)
        }
        Thread.sleep(3000)
        expect:
        name1 == "james"
        name2 == "james"
    }

二级缓存的特点

二级缓存是Mapper级别的,RC 下换一个mapper查询就可以避免缓存

def "证明不同的mapper, 用的是不一样的二级缓存"() {
        given:
        mapper.insert(1, "james")
        def name1
        def name2
        transaction.execute {
            // 当前线程查询
            name1 = mapper.selectNameByIdWithoutL1Cache(1)

            // 新开线程更新并提交
            executor.execute {
                transaction.execute {
                    mapper.updateNameById("kobe", 1)
                }
            }

            // 确保更新线程提交成功
            Thread.sleep(2000)

            // 使用其他mapper排查缓存的干扰
            name2 = anotherMapper.selectNameById(1)
        }
        Thread.sleep(3000)
        expect:
        name1 == "james"
        name2 == "kobe"
    }

后记

MyBatis 的缓存在RR隔离级别暂时没发现什么不一致问题。目前经历过的项目都是RR级别,没有看到MyBatis缓存相关的设置。这次探索的主要收益还是在 Spock 及 Groovy 精炼的语法上面。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/156273.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

insert唯一索引加锁流程

MySQL版本&#xff1a;8.0.29 测试表&#xff1a; 测试数据&#xff1a; 开始测试&#xff1a; 事务1执行 加锁分析&#xff1a; mysql> SELECT * FROM performance_schema.data_locks\G *************************** 1. row ***************************ENGINE: INNO…

Express 中使用JWT进行登录验证

cookie 篇 : Node.js 中 cookie的验证登录 | session 篇 : Node.js 中 session验证登录 在前面讲过了两种验证登录的方式&#xff0c;其一是cookie&#xff0c;其二是session&#xff1b;那么在讲JWT之前先来简单的回顾这两种方式区别&#xff1b;cookie和sessi…

Java笔记023-包装类、String类、字符串的特性、String类、StringBuffer类、StringBuilder类

常用类包装类包装类的分类1、针对八种基本定义相应的引用类型-包装类2、有了类的特点&#xff0c;就可以调用类中的方法基本数据类型包装类booleanBooleancharCharacterbyteByteshortShortintIntegerlongLongfloatFloatdoubleDouble包装类和基本数据的转换演示包装类和基本数据…

LInux目录结构

文章目录Linux的目录结构Linux的目录结构Linux路径的描述方式目录各功能介绍HOME目录和工作目录Linux的目录结构 Linux的目录结构 Linux的目录结构是一个树型结构。 Windows 系统可以拥有多个盘符, 如 C盘、D盘、E盘 Linux没有盘符这个概念, 只有一个根目录 /, 所有文件都在…

【UE4 第一人称射击游戏】37-拾取副武器“M4A1”

上一篇&#xff1a;【UE4 第一人称射击游戏】36-切换武器时改变UI本篇效果&#xff1a;在拾取副武器“M4A1”前&#xff0c;点击键盘2键是无法切换武器的&#xff0c;当拾取武器后&#xff0c;点击键盘2键可以切换武器&#xff08;目前仅是右下角的图标和文字实现了切换&#x…

VBA小模板,一个不放回的抽奖用的例子

1 问题 一个不放回的抽奖用VBA怎么写&#xff0c;下面用一个类似对对碰/ 翻牌子的游戏&#xff08;抽到的奖励不放回&#xff0c;可抽的东西越来越少&#xff09;来举例说明 1.1 首先要回顾下几个经典的随机模型 古典概型&#xff0c;重点就是每次抽奖的各个奖品&#xff0c;概…

宇视门禁一体机接线图

宇视门禁一体机接线图宇视门禁一体机带反馈信号电磁锁接线图门禁的GND线通用&#xff0c;可以连到同一根线上&#xff0c;也可以分开连简单整理如图一体机线颜色和功能对应表颜色线路绿色RS485_A粉色RS485_B黑色GND蓝色WIEGAND_OUT_D0白色WIEGAND_OUT_D1灰色WIEGAND_IN_D0棕色W…

测试面试真题|工作2年,从小厂到大厂,薪资翻倍是怎样的体验?

最近&#xff0c;霍格沃兹测试学院学员 C 同学成功拿下某互联网大厂年薪 30W 测试开发岗位 Offer&#xff0c;顺利完成从手工测试工程师到测试开发的逆袭&#xff0c;薪资翻倍&#xff08;涨幅 100%&#xff09;&#xff0c;并获得了学院颁发的优秀学员奖学金。C 同学工作刚满 …

【3D目标检测】Delving into Localization Errors for Monocular 3D Object Detection

目录概述细节错误分析概述 本文是基于单目图像的3D目标检测方法。 【2021】【MonoDLE】 研究的问题: 核心问题&#xff1a;如何提高基于单目图像的3D目标检测的效果。作者量化了每个子任务的整体影响&#xff0c;观察到以下现象 观察一&#xff1a;定位误差是制约目标检测性…

变异凯撒题解

题目变异凯撒&#xff0c;说明没有使用一般的凯撒加密看到一个密码&#xff0c;我们可以找一下规律首先密文不完全是字母&#xff0c;但是经典凯撒加密的密文一定都是字母&#xff0c;说明这个题目可能是ASCLL码表偏移&#xff0c;而不是字母表偏移经典凯撒加密是字母和字母的偏…

SSM纯注解后台代码整合(Spring+SpringMvc+Mybatis)

SSM后台整合&#xff08;SpringSpringMvcMybtis事务Rest风格统一结果封装统一异常处理拦截器&#xff09; 文章目录1 基础环境搭建1.1 建表1.2 创建web项目1.3 导入依赖坐标&#xff08;pom.xml&#xff09;1.4 包路径的创建1.5 在pojo包下编写book实体类1.6 在webapp包下导入静…

很多人还不知道中视频计划手机上发布多端横竖屏视频的方法

如果说你刚开始接触中视频&#xff0c;你必须要学会的小知识。 横屏视频是16&#xff1b;9的视频&#xff0c;一般是手机横向拍摄的视频。 上传这样的视频有两种方法。第一种是需要用到电脑&#xff0c;第二种就是我今天要分享的这种&#xff0c;没有电脑&#xff0c;我们用手…

测试新人入职第一天都做什么?

测试入职第一天都做什么&#xff1f; 大家都知道&#xff0c;入职一家新公司就是一个新的起点&#xff0c;新的开始。不管在之前公司干了多久&#xff0c;还是第一次踏入社会进入企业上班&#xff0c;都需要遵守新公司的规则以及规章制度。不管企业的规章制度是什么&#xff0…

前端提交信息规范 commit规范 commitlint husky commitizen

前端提交规范 约定式提交约定式提交安装commitizen 及其适配器husky commitlint 提交校验至此恭喜你已经配置好提交校验了&#xff0c;快去试试吧技术同学开发中有没有出现 &#xff1a; 因某种原因当需要回滚&#xff0c;这时候一看之前提交的massage真的乌烟瘴气 无法分辨回…

使用ResNet50实现CIFAR100数据集的训练

如果对你有用的话&#xff0c;希望能够点赞支持一下&#xff0c;这样我就能有更多的动力更新更多的学习笔记了。&#x1f604;&#x1f604; 使用ResNet进行CIFAR-10数据集进行测试&#xff0c;这里使用的是将CIFAR-10数据集的分辨率扩大到32X32&#xff0c;因为算力相关的…

蓝桥杯2019年第十届省赛C++B组

文章目录A&#xff1a;组队&#xff08;5分 √&#xff09;B&#xff1a;年号字串&#xff08;5分 √&#xff09;C: 数列求值&#xff08;10分 √&#xff09;D: 数的分解&#xff08;10分 &#xff09;F: 特别数的和&#xff08;15分 √&#xff09;A&#xff1a;组队&#x…

【蓝桥杯简单篇】Python组刷题日寄Part05

刷题日记&#xff1f;刷题日寄&#xff01; 萌新备战蓝桥杯python组 &#x1f339; 发现有需要纠正的地方&#xff0c;烦请指正&#xff01; &#x1f680; 欢迎小伙伴们的三连关注&#xff01; 往期系列&#xff1a; 【蓝桥杯简单篇】Python组刷题日寄Part01 【蓝桥杯简单篇】…

Oracle Mysql审计日志等保测评

mysql oracle的审计日志 mysql的审计日志说是有两种方法&#xff0c;一种是需要安装插件模式的审计&#xff0c;一种直接开一个参数 1.安装插件模式 一、 mysql 日志 配置永久配置 &#xff1a; 保存时间及大小 https://blog.csdn.net/m0_51197424/article/details/12432840…

设计模式(二)----软件设计原则

在软件开发中&#xff0c;为了提高软件系统的可维护性和可复用性&#xff0c;增加软件的可扩展性和灵活性&#xff0c;要尽量根据7条原则来开发程序&#xff0c;从而提高软件开发效率、节约软件开发成本和维护成本。 1、单一职责原则 ( 核心&#xff1a;尽量保证类&#xff0…

毫米波雷达和视觉融合的学习路线

了解各个传感器的成像原理&#xff0c;知其所以然&#xff0c;同时了解每种传感器的对比及优缺点&#xff0c;为什么要用这几种融合&#xff0c;可以通过去看一些德州仪器的雷达原理视频&#xff08;b站&#xff09;&#xff0c;雷达工作手册等。先广而后深&#xff1a;了解经典…