详解Mysq MVCC多版本的并发控制


Posted in MySQL onApril 29, 2022

1、MVCC

MVCC,全称Multi-Version Concurrency Control,即多版本并发控制。MVCC是一种并发控制的方法,一般在数据库管理系统中,实现对数据库的并发访问,在编程语言中实现事务内存。

MVCC在MySQL InnoDB中的实现主要是为了提高数据库并发性能,用更好的方式去处理读写冲突,做到即使有>读写冲突时,也能做到不加锁,非阻塞并发读。

2、当前读

像select lock in share mode(共享锁), select for update ; update, insert ,delete(排他锁)这些操作都是一种当前读,为什么叫当前读?就是它读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。

3、快照读(提高数据库的并发查询能力)

像不加锁的select操作就是快照读,即不加锁的非阻塞读;快照读的前提是隔离级别不是串行级别,串行级别下的快照读会退化成当前读;之所以出现快照读的情况,是基于提高并发性能的考虑,快照读的实现是基于多版本并发控制,即MVCC,可以认为MVCC是行锁的一个变种,但它在很多情况下,避免了加锁操作,降低了开销;既然是基于多版本,即快照读可能读到的并不一定是数据的最新版本,而有可能是之前的历史版本

4、当前读、快照读、MVCC关系

MVCC多版本并发控制指的是维持一个数据的多个版本,使得读写操作没有冲突,快照读是MySQL为实现MVCC的一个非阻塞读功能。MVCC模块在MySQL中的具体实现是由三个隐式字段,undo日志、read view三个组件来实现的。

5、MVCC实现原理

mvcc的实现原理主要依赖于记录中的三个隐藏字段,undolog,read view来实现的。

隐藏字段

行记录除了我们自定义的字段外,还有数据库隐式定义的DB_TRX_ID,DB_ROLL_PTR,DB_ROW_ID等字段

DB_TRX_ID

6字节,最近修改事务id,记录创建这条记录或者最后一次修改该记录的事务id

DB_ROLL_PTR

7字节,回滚指针,指向这条记录的上一个版本,用于配合undolog,指向上一个旧版本

DB_ROW_JD

6字节,隐藏的主键,如果数据表没有主键,那么innodb会自动生成一个6字节的row_id

undo log

undolog被称之为回滚日志,表示在进行insert,delete,update操作的时候产生的方便回滚的日志 当进行insert操作的时候,产生的undolog只在事务回滚的时候需要,并且在事务提交之后可以被立刻丢弃 当进行update和delete操作的时候,产生的undolog不仅仅在事务回滚的时候需要,在快照读的时候也需要,所以不能随便删除,只有在快照读或事务回滚不涉及该日志时,对应的日志才会被purge线程统一清除(当数据发生更新和删除操作的时候都只是设置一下老记录的deleted_bit,并不是真正的将过时的记录删除,因为为了节省磁盘空间,innodb有专门的purge线程来清除deleted_bit为true的记录,如果某个记录的deleted_id为true,并且DB_TRX_ID相对于purge线程的read view 可见,那么这条记录一定时可以被清除的)

Read View

Read View是事务进行快照读操作的时候生产的读视图,在该事务执行快照读的那一刻,会生成一个数据系统当前的快照,记录并维护系统当前活跃事务的id,事务的id值是递增的。

6、MVCC核心思想

MVCC 的核心思想是: 我可以查到在我这个事务开始之前已经存在的数据,即使它在后面被修改或者删除了。在我这个事务之后新增的数据,我是查不到的。

MVCC查找规则:只能查找创建时间小于等于当前事务ID的数据和删除时间大于当前事务ID的行(或未删除)

详解Mysq MVCC多版本的并发控制

如图,在Transaction1事务中插入两条数据,并提交事务,然后在Transaction2事务中读取,读取到两条数据

详解Mysq MVCC多版本的并发控制

如图,在Transaction3事务中插入一条为老连的数据,然后在Transaction2事务读取,根据mvcc规则,不能查到在我的事务开始之后插入插入的数据,老连的创建ID大于2,所以只能查到两条数据

详解Mysq MVCC多版本的并发控制

如图,在Transaction4事务中删除id为2的数据,然后在Transaction2事务读取,根据mvcc规则,可以查到在我的事务开始之后插删除的数据,老晁还是可以查出来所以还是查到两条数据

详解Mysq MVCC多版本的并发控制

如图,在Transaction5事务中下,添加一条name=涛哥的数据,删除id=1数据,修改name=涛哥的id为1,然后在Transaction2事务读取,根据mvcc规则,可以查到在我的事务开始之后插删除的数据,老严还是可以查出来,所以还是查到两条数据

通过以上演示我们能看到,通过版本号的控制,无论其他事务是插入、修改、删除,Transaction2事务查询到的数据都没有变化。

到此这篇关于Mysql MVCC多版本并发控制详情的文章就介绍到这了!


Tags in this post...

MySQL 相关文章推荐
浅析InnoDB索引结构
Apr 05 MySQL
MySQL infobright的安装步骤
Apr 07 MySQL
Mysql服务添加 iptables防火墙策略的方案
Apr 29 MySQL
修改MySQL的数据库引擎为INNODB的方法
May 26 MySQL
MySQL完整性约束的定义与实例教程
May 30 MySQL
MySQL为id选择合适的数据类型
Jun 07 MySQL
解决mysql问题:由于找不到MSVCR120.dll,无法继续执行代码
Jun 26 MySQL
python中的mysql数据库LIKE操作符详解
Jul 01 MySQL
QT连接MYSQL数据库的详细步骤
Jul 07 MySQL
Mysql外键约束的创建与删除的使用
Mar 03 MySQL
mysql的单列多值存储实例详解
Apr 05 MySQL
MySql如何将查询的出来的字段进行转换
Jun 14 MySQL
详解Mysql事务并发(脏读、不可重复读、幻读)
MySQL 条件查询的常用操作
Apr 28 #MySQL
mysql 子查询的使用
Apr 28 #MySQL
Mysql 数据库中的 redo log 和 binlog 写入策略
Apr 26 #MySQL
优化Mysql查询的示例
Apr 26 #MySQL
MySQL的存储过程和相关函数
Apr 26 #MySQL
mysql 索引的数据结构为什么要采用B+树
You might like
基于PHP创建Cookie数组的详解
2013/07/03 PHP
php实现的click captcha点击验证码类实例
2014/09/23 PHP
phpmyadmin中禁止外网使用的方法
2014/11/04 PHP
Javascript实例教程(19) 使用HoTMetal(5)
2006/12/23 Javascript
Javascript valueOf 使用方法
2008/12/28 Javascript
Prototype Function对象 学习
2009/07/12 Javascript
utf-8编码引起js输出中文乱码的解决办法
2010/06/23 Javascript
基于jQuery实现点击弹出层实例代码
2016/01/01 Javascript
jquery ajax双击div可直接修改div中的内容
2016/03/04 Javascript
angularjs使用directive实现分页组件的示例
2017/02/07 Javascript
Vue实现点击时间获取时间段查询功能
2020/08/21 Javascript
基于Koa(nodejs框架)对json文件进行增删改查的示例代码
2019/02/02 NodeJs
Vue 中获取当前时间并实时刷新的实现代码
2020/05/12 Javascript
[02:14]DOTA2英雄基础教程 修补匠
2013/12/23 DOTA
[04:19]DOTA2完美大师赛第四天精彩集锦
2017/11/26 DOTA
[01:04:22]2018DOTA2亚洲邀请赛 3.31 小组赛 B组 IG vs EG
2018/04/01 DOTA
[01:04:48]VGJ.S vs TNC Supermajor 败者组 BO3 第一场 6.6
2018/06/07 DOTA
python里对list中的整数求平均并排序
2014/09/12 Python
Python SqlAlchemy动态添加数据表字段实例解析
2018/02/07 Python
Python爬虫抓取代理IP并检验可用性的实例
2018/05/07 Python
python日志logging模块使用方法分析
2019/05/23 Python
Python3内置模块random随机方法小结
2019/07/13 Python
Django之PopUp的具体实现方法
2019/08/31 Python
Python从列表推导到zip()函数的5种技巧总结
2019/10/23 Python
Python爬虫抓取指定网页图片代码实例
2020/07/24 Python
python 利用zmail库发送邮件
2020/09/11 Python
博朗(Braun)俄罗斯官方商店:德国小家电品牌
2019/09/24 全球购物
Claire’s法国:时尚配饰、美容、珠宝、头发
2021/01/16 全球购物
MySQL面试题目集锦
2016/04/14 面试题
践行三严三实心得体会
2014/10/13 职场文书
校园音乐节目广播稿
2015/08/19 职场文书
写自招自荐信的绝招!
2019/04/19 职场文书
解决Jupyter-notebook不弹出默认浏览器的问题
2021/03/30 Python
新手必备之MySQL msi版本下载安装图文详细教程
2021/05/21 MySQL
Python极值整数的边界探讨分析
2021/09/15 Python
Python 数据可视化之Matplotlib详解
2021/11/02 Python