【MySQL】MVCC机制(undo log,read view)

文章目录

  • 前言
  • 一. 预备知识
  • 二. 模拟MVCC
  • 三. Read View
  • 四. RC与RR的本质区别
  • 结束语

前言

MVCC(多版本并发控制)是一种用来解决读-写冲突的无锁并发控制

MVCC为事务分配单向增长的事务ID,为每个修改保存一个版本,版本与事物ID相关联,读操作只读该事务开始前的数据库的快照,所以MVCC可以解决以下问题

  • 在并发读写数据库数据时,读操作不用阻塞写操作,写操作也不用阻塞读操作,提高了数据库并发读写性能
  • 同时解决脏读,幻读,不可重复读等事务隔离性问题,但不能解决更新丢失问题

数据库并发的场景

数据库并发场景有三种

  1. 读-读:不存在任何问题,不需要并发控制,但有使用共享锁
  2. 读-写:有线程安全问题,可能会造成事务隔离性问题,如脏读,幻读,不可重复读
  3. 写-写:有线程安全问题,可能会存在更新丢失问题,如第一类丢失更新问题和第二类更新丢失问题

补充

第一类更新丢失问题(回滚丢失,Lost update)

第一类更新丢失是指,一个事务被撤销,可能导致其他事务已提交的更新数据被覆盖

时间序号事务一事务二
T1begin开启事务begin开启事务
T2查询余额money=1000查询余额money=1000
T3存款100,money=1100
T4取款100,money=900
T5commit提交事务
T6回滚取款操作,money恢复1000

正如上述事例,事务一二开始查询余额都是1000,事务二先进行存款操作,并提交。
事务一不知道事务二的存在,进行取款操作,但是又进行了回滚,就会将余额恢复成最开始查询的数额,这就覆盖了事务二的更新操作

第二类更新丢失问题(覆盖丢失/两次更新问题,Second lost update)

第二类更新丢失是指,当两个事物或多个事务查询相同的记录,然后各自基于查询结果更新数据

时间序号事务一事务二
T1begin开启事务begin开启事务
T2查询余额money=1000查询余额money=1000
T3取款100,money=900
T4commit提交事务
T5存款100,money=1100
T6commit提交事务

事务一二查询相同余额=1000,事务二先进行取款操作,money=900,但事务一后续基于自己的查询结果,进行存款操作,money=1100,这就覆盖了事务二的数据更新

一. 预备知识

学习MVCC前,我们要有以下三个知识了解

  • 3个记录隐藏字段
  • undo日志(undo log)
  • Read View

3个记录隐藏字段

这3个字段是记录信息

  • DB_TRX_ID:6byte。最近修改该记录的事务ID,记录创建这条记录/最后一次修改改记录的事务ID
  • DB_ROLL_PTR:7byte。回滚指针,指向这条记录的上一个版本(指向历史版本,历史版本在undo log中)
  • DB_ROW_ID:6byte。隐含的自增ID隐藏主键),如果数据表没有主键,InnoDB会自动以DB_ROW_ID产生一个聚簇索引

补充:实际还有一个标记删除/更新的flag字段,在事务中删除记录,会将该flag字段标记为删除

比如如下学生表,有name和age两个属性

mysql> select * from student;
+--------+-----+
| name   | age |
+--------+-----+
| 张三   |  28 |
+--------+-----+

但其实还有3个隐藏字段

nameageDB_TRX_IDDB_ROW_IDDB_ROLL_PTR
张三28最后修改该记录的事务ID隐式主键 1回滚指针(指向历史记录)

undo log

MySQL是网络进程服务,所有的索引,事务,隔离性,日志等,都是在内存中完成的,即在MySQL内部的相关缓冲区中保存数据,再在合适的时候,进行刷盘,将数据写入磁盘,达到持久化
所以,undo log简单理解,就是MySQL中的一段内存缓冲区,用来保存日志数据

在数据库事务开始之前,MySQL会将记录保存在undo log中,如果事务回滚或者数据库崩溃时,可以利用undo log日志中记录的日志信息进行回退。同时也可以提供多版本并发控制下的读(MVCC

undo log的生命周期

undo log产生:在事务开始之前生成
undo log销毁:当事务提交之后,undo log并不能马上删除,而是放入待清理的链表,由purge线程判断是否有其他事务在使用undo log保存的上一个事务之前的版本信息,决定是否可以清理undo log的日志空间
注意:undo log也会产生redo log,undo log也需要持久化保护

undo log和redo log的区别

undo log是逻辑日志,实现事务的原子性

  1. undo log记录的是事务开始前的数据状态,记录的是更新前的值
  2. undo log实现事务的原子性提供回滚

redo log是物理日志,实现事务的持久性

  1. redo log记录的是事务完成后的数据状态,记录的是更新后的值
  2. redo log实现事务的持久性(保证数据的完整性)

Read view稍后再讲解,因为需要快照这一概念

二. 模拟MVCC

假设现在有一个事务,其事务ID为10,对student表中记录进行修改update:将name(张三)改成name(李四)

MVCC过程如下:

  • 因为是修改,所以要给该记录加行锁
  • 修改前,先将原本的数据拷贝一份到undo log中,相当于undo log中有一个备份(写时拷贝
  • 然后,MySQL相当于有两行相同的记录,修改是修改原始记录的name,并且修改原始记录的隐藏字段DB_TRX_ID为修改该数据的事务ID,即10,而该记录的回滚指针DB_ROLL_PTR,里面写入undo log中副本数据的地址,表示上一个版本
  • 事务10提交,释放锁

结果如下图

在这里插入图片描述

此时,最新的记录就是name='李四’的那条

接着,又有一个事务11要对student表进行修改(update):将age(28)改成age(38)

  • 因为是修改,所以需要给该记录加行锁
  • 修改前,拷贝一份原始数据到undo log中
  • 将原始数据的age(28)改成age(38),并且修改DB_TRX_ID为事务11ID,DB_ROLL_PTR指向undo log中的备份数据地址,表示上一个版本数据

结果如下图

在这里插入图片描述

如此就形成了一个基于链表记录的历史版本链。回滚其实就是利用历史数据,覆盖当前数据
上述的一个个版本,被称为一个个快照

update可以形成版本,delete和insert同样也可以。

delete删除数据是设置flag字段为删除,回滚只要再修改flag字段即可

insert插入数据,虽然没有历史版本,但是为了回滚操作,insert的数据也会被放入undo log中,如果当前事务commit提交了,那么undo log的历史insert记录就会被清空


有了undo log,select读取就被分为了两种读:

  1. 快照读,读取历史版本
  2. 当前读,读取最新数据,select lock in share mode(共享锁),select for update。增删改也是读取当前数据

当有多个事务同时增删改时,都是当前读,势必需要加锁,此时select如果也是当前读,那就会被阻塞,这就是串行化
但如果是快照读,读取历史版本,则不受加锁限制,可以并发运行,这就是MVCC的意义。

隔离级别决定了select是当前读还是快照读
事务总是有先有后,而事务可以分为三个阶段:执行前,执行中,执行后
隔离性的目的就是让不同的事务看到它该看到的内容

三. Read View

如何实现隔离级别呢?其实就是实现了Read View

Read View是事务进行快照读操作时产生的一个读视图,在该事务执行的快照读的那一刻,会生成数据库系统当前的一个快照,记录并维护系统当前活跃事务的ID(每个事务开始时,都会被分配到一个ID,此ID是自增的,事务越新,ID值越大)

Read View在MySQL源码中,是一个。本质是用来进行可见性判断的。即当我们某个事务执行快照读时,对该记录创建一个Read View读视图,以此判断当前事务能够看到哪个版本的数据,既可能是当前最新数据,也可能是该记录在undo log里的某个历史版本数据

比较关键的属性如下:

class ReadView
{
...
private:
trx_id_t m_low_limit_id;
trx_id_t m_up_limit_id;
trx_id_t m_creator_trx_id;
ids_t m_ids;
bool m_closed;
...
}
  • m_ids:创建视图时的活跃事务id列表
  • m_low_limit_id:翻译为高水位,生成ReadView时,系统尚未分配的下一个事务ID,也就是目前已有的事务ID的最大值+1,大于等于这个ID的事务均不可见
  • m_up_limit_id:翻译为低水位,记录m_ids列表中事务ID的最小ID,小于这个ID的事务均可见
  • m_creator_trx_id:创建该读视图的事务ID

我们在实际读取数据版本链的时候,能读取到每一个版本对应的事务ID,也就是隐藏字段DB_TRX_ID
而通过DB_TRX_ID和以上四个属性作比较,就可以判断该记录是否应该被读取到

在这里插入图片描述

m_ids列表记录着形成快照的时,活跃的事务ID

  1. 如果记录中的DB_TRX_ID,和m_up_limit_id,即m_ids中最小的事务ID作比较,小于这个事务ID,代表该事务一定已经提交,其记录一定是历史数据可以读取
  2. 如果记录中的DB_TRX_ID,等于m_creator_trx_id,代表是自己修改的数据可以读取
  3. 如果记录中的DB_TRX_ID,在m_ids中,代表修改该记录的事务还未提交或在形成快照后才提交不可读取
  4. 如果记录中的DB_TRX_ID,大于等于m_low_limit_id,即在快照形成时,系统还未分配的事务ID,代表该数据是在快照形成后才形成的,不可读取

如果查找不应该看的版本,可以按照回滚指针,跳转到上一个历史版本,直到符合条件


模拟Read View过程

假设当前有记录;

nameageDB_TRX_IDDB_ROW_IDDB_ROLL_PTR
张三28null1null

目前不关心创建该记录的事务ID,并且因为是创建的记录,所以没有历史版本,所以回滚指针为null

事务操作:

事务1[id=1]事务2[id=2]事务3[id=3]事务4[id=4]
beginbeginbeginbegin
修改且提交
进行中快照读进行中

事务4:修改name(张三)变成name(李四)

当事务2对某行数据进行快照读时,数据库会为该行数据生成一个Read View读视图

事务2的Read View
m_ids:1,3
up_limit_id:1
low_limit_id:4+1=5,读视图生成时,系统尚未分配的下一个事务ID
creator_trx_id:2

此时的版本链如下:

在这里插入图片描述

因为事务4在事务2形成快照前就提交了,所以是可见的

事务2在快照读时,就会拿该记录的DB_TRX_ID跟Read View中的几个属性比较,判断该版本是否可见

比较步骤
DB_TRX_ID(4)< up_limit_id(1)? 不小于,下一步
DB_TRX_ID(4)>= low_limit_id(5) ? 不大于,下一步
m_ids.contains(DB_TRX_ID) ? 不包含,说明事务4不在当前的活跃事务中。

四. RC与RR的本质区别

RC即Read Committed(读提交),RR即Repeatable Read(可重复读)
详细定义可见【MySQL】事务

Read View生成时机的不同,从而造成RC,RR级别下快照读的结果不同

RR级别的快照读

在RR级别下的某个事务对某条记录的第一次快照读会创建一个快照和Read View,将当前系统活跃的其他事务记录起来
之后再快照读时,还是使用同一个Read View,所以只要当前事务在其他事务提交之前使用过快照读,那么之后的快照读使用的都是同一个Read View,对之后的修改不可见

即在RR界别下,快照读生成Read View时,Read View会记录此时所有其他活动事务的快照,这些事务的修改对于当前事务都是不可见的,而早于Read View创建的事务所做的修改均是可见的

RC级别的快照读

在RC级别下,事务没词快照读都会新生成一个快照和Read View,所以即使后来的事务提交了,其修改结果也可见,因为RC级别下的Read View是每次快照读都会新形成的

RC级别下的Read View是每次快照读都会新形成,而RR级别的Read View只会在第一次快照读时形成

推荐文章
【MySQL笔记】正确的理解MySQL的MVCC及实现原理
详细分析MySQL事务日志(redo log和undo log)
【MySQL】InnoDB 如何避免脏读和不可重复读

结束语

感谢看到此处
如果觉得本篇文章对你有所帮助的话,不妨点个赞支持一下博主,拜托啦,这对我真的很重要。
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

74X138元件怎么找——错误解决方法

1.在做74X138的时候根据课本&#xff0c;无法在现有的库中找到74X138&#xff0c;搜索了老师发的库中&#xff0c;都是集成库打不开&#xff0c;那我该怎么办? 根据这个课本P343&#xff0c;&#xff08;即机械工业出版社&#xff0c;刘超&#xff0c;包建荣&#xff0c;俞优姝…

CTF-Reverse---VM虚拟机逆向[HGAME 2023 week4]vm题目复现【详解】

文章目录 前言0x1[HGAME 2023 week4]vm提取汇编指令mov指令push指令&pop指令运算操作cmp指令jmp指令je 和 jne exp 前言 没有前言。终于搞定第二题了。费劲是真的费。 题目在nssctf上有。 这一题写完才看到有提示&#xff0c;是关于构建结构体的&#xff0c;下次试试。 0x…

1360. 日期之间隔几天

1360. 日期之间隔几天 Java代码&#xff1a; 【DateFormat】DateFormat用于实现日期的格式化 import java.text.DateFormat; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; // 好像已过时class Solution {public int daysBet…

vue3+ts 提取公共方法

因为好多页面都会使用到这个效验规则&#xff0c;封装一个校检规则&#xff0c;方便维护 封装前 封装后

【机器学习】四、计算学习理论

1 基础知识 计算学习理论&#xff08;computational learning theory&#xff09;&#xff1a;关于通过“计算”来进行“学习”的理论&#xff0c;即关于机器学习的理论基础&#xff0c;其目的是分析学习任务的困难本质&#xff0c;为学习算法体统理论保证&#xff0c;并根据结…

frp-内网穿透部署-ubuntu22服务器-windows server-详细教程

文章目录 1.下载frp2.Ubuntu修改root用户3.配置服务器3.1.配置frps.ini文件3.2.设置服务文件3.3.设置开机自启和服务操作3.4.后台验证3.5.服务器重启 4.配置本地window4.1.frpc配置4.2.添加开机计划启动4.3.控制台启动隐藏窗口 5.centos防火墙和端口5.1.开放端口5.2.查看端口 6…

【电路笔记】-相量图和相量代数

相量图和相量代数 文章目录 相量图和相量代数1、概述2、相量图3、相量代数3.1 加减3.2 差异化与整合 4、总结 1、概述 交流电信号可以用三种不同的方法来表示&#xff0c;以便表征和实现代数运算。 前面的文章中已经介绍了两种方法&#xff0c;本文稍后将介绍一种新的图形方法…

Pytorch 注意力机制解析与代码实现

目录 什么是注意力机制1、SENet的实现2、CBAM的实现3、ECA的实现4、CA的实现 什么是注意力机制 注意力机制是深度学习常用的一个小技巧&#xff0c;它有多种多样的实现形式&#xff0c;尽管实现方式多样&#xff0c;但是每一种注意力机制的实现的核心都是类似的&#xff0c;就…

【计算机网络】(谢希仁第八版)第一章课后习题答案

1.计算机网络可以向用户提供哪些服务&#xff1f; 答&#xff1a;例如音频&#xff0c;视频&#xff0c;游戏等&#xff0c;但本质是提供连通性和共享这两个功能。 连通性&#xff1a;计算机网络使上网用户之间可以交换信息&#xff0c;好像这些用户的计算机都可以彼此直接连…

036-第三代软件开发-系统时间设置

第三代软件开发-系统时间设置 文章目录 第三代软件开发-系统时间设置项目介绍系统时间设置演示效果QML 实现小伙伴自创 TumblerQt 家 Tumbler C 端实现 总结一下 关键字&#xff1a; Qt、 Qml、 Time、 时间、 系统 项目介绍 欢迎来到我们的 QML & C 项目&#xff01;…

【VPX630】青翼 基于KU115 FPGA+C6678 DSP的6U VPX通用超宽带实时信号处理平台

板卡概述 VPX630是一款基于6U VPX总线架构的高速信号处理平台&#xff0c;该平台采用一片Xilinx的Kintex UltraScale系列FPGA&#xff08;XCKU115&#xff09;作为主处理器&#xff0c;完成复杂的数据采集、回放以及实时信号处理算法。采用一片带有ARM内核的高性能嵌入式处理器…

Nginx负载均衡 以及Linux前后端项目部署

一、Nginx简介 Nginx是一款高性能的开源Web服务器和反向代理服务器。它由俄罗斯的程序设计师Igor Sysoev创建&#xff0c;旨在解决传统Web服务器的性能限制问题。 Nginx采用事件驱动的架构和异步非阻塞的处理方式&#xff0c;使其能够处理大量并发连接&#xff0c;并具备良好…

【分享】winterm ssh登录报错Unkown error

非软文哈&#xff0c;实测Winterm非常好用&#xff0c;唯一的障碍是 某些特定服务器ssh登录报错Unkown error 后经github issue得知&#xff0c;关闭会话设置-ssh选项卡中的 尝试键盘交互认证的勾即可。 https://github.com/kingToolbox/WindTerm/issues/1922

二叉树题目:在二叉树中增加一行

文章目录 题目标题和出处难度题目描述要求示例数据范围 解法一思路和算法代码复杂度分析 解法二思路和算法代码复杂度分析 题目 标题和出处 标题&#xff1a;在二叉树中增加一行 出处&#xff1a;623. 在二叉树中增加一行 难度 5 级 题目描述 要求 给定一个二叉树的根结…

Vue 数据绑定 和 数据渲染

目录 一、Vue快速入门 1.简介 : 2.MVVM : 3.准备工作 : 二、数据绑定 1.实例 : 2.验证 : 三、数据渲染 1.单向渲染 : 2.双向渲染 : 一、Vue快速入门 1.简介 : (1) Vue[/vju/]&#xff0c;是Vue.js的简称&#xff0c;是一个前端框架&#xff0c;常用于构建前端用户…

C++二分查找算法的应用:俄罗斯套娃信封问题

本文涉及的基础知识点 二分查找 题目 给你一个二维整数数组 envelopes &#xff0c;其中 envelopes[i] [wi, hi] &#xff0c;表示第 i 个信封的宽度和高度。 当另一个信封的宽度和高度都比这个信封大的时候&#xff0c;这个信封就可以放进另一个信封里&#xff0c;如同俄罗…

assert断言与const修饰指针的妙用(模拟实现strcpy函数)

assert断言 目录 assert断言的妙用&#xff1a; 头文件&#xff1a; 使用方法&#xff1a; const修饰指针的妙用 主要用法 const在*左边 const在*右边 断言和const修饰指针的应用 模拟实现C语言strcpy函数 1、若字符串str1,str2有空指针怎么办&#xff1f; 2.str2改变…

【Unity实战】最全面的库存系统(一)

文章目录 先来看看最终效果前言定义物品定义人物背包物品插槽数据拾取物品物品堆叠绘制UI移动拖拽物品选中物品跟随鼠标移动背包物品交换物品拆分物品物品堆叠完结先来看看最终效果 前言 它又来了,库存系统我前面其实一句做过很多次了,但是这次的与以往的不太一样,这个将是…

微信小程序:两层循环的练习,两层循环显示循环图片大图(大图显示、多层循环)

效果 代码分析 外层循环 外层循环的框架 <view wx:for"{{info}}" wx:key"index"></view> wx:for"{{info}}"&#xff1a;这里wx:for指令用于指定要遍历的数据源&#xff0c;即info数组。当遍历开始时&#xff0c;会依次将数组中的每…

前端架构体系调研整理汇总

1.公司研发人数与前端体系 小型创业公司 前端人数&#xff1a; < 3 人 产品类型&#xff1a; 产品不是非常成熟&#xff0c;比较新颖。 项目流程&#xff1a;不完善&#xff0c;快、紧促&#xff0c;没有固定的时间排期。 技术栈&#xff1a; 没有历史包袱&#xff0c;技…
最新文章