详解MySQL事务的隔离级别与MVCC


Posted in MySQL onApril 22, 2021

事务隔离级别

事务并发执行遇到的问题

  • 脏写
    • 如果一个事务修改了另一个未提交事务修改过的数据,那就意味着发生了脏写
  • 脏读
    • 如果一个事务读到了另一个未提交事务修改过的数据,那就意味着发生了脏读
  • 不可重复读
    • 如果一个事务只能读到另一个已经提交的事务修改过的数据,并且其他事务每对该数据进行一次修改并提交后,该事务都能查询到最新值,那就意味着发生了不可重复读
  • 幻读
    • 如果一个事务先根据某些条件查询出一些记录,之后另一个事务又向表中插入了符合这些条件的记录,原先的事务再次按照该条件查询时,能把另一个事务插入的记录也读出来,那就意味着发生了幻读。
    • 幻读强调的是一个事务按照某个相同条件多次读取记录时,后读取时读到了之前没有读到的记录
    • 那对于先前已经读到的记录,之后又读取不到这种情况,算啥呢?其实这相当于对每一条记录都发生了不可重复读的现象。幻读只是重点强调了读取到了之前读取没有获取到的记录。

SQL标准中的四种隔离级别

  • READ UNCOMMITTED: 未提交读 脏读、不可重复读、幻读 发生
  • READ COMMITTED:已提交读 不可重复读、幻读 发生
  • REPEATBLE READ:可重复读 幻读 发生
  • SERIALIZABLE:可串行化 不发生

MySQL中支持的四种隔离级别

  • MySQL在REPEATABLE READ隔离级别下,是可以禁止幻读问题的发生的(关于如何禁止我们之后会详细说明的)
  • MySQL默认隔离级别为REPEATABLE READ

MVCC原理

版本链

对于使用InnoDB存储引擎的表来说,它的聚簇索引记录中都包含两个必要的隐藏列

  • trx_id:每次一个事务对某条聚簇索引记录进行改动时,都会把该事务的事务id赋值给trx_id隐藏列
  • roll_pointer:每次对某条聚簇索引记录进行改动时,都会把旧的版本写入到undo日志,然后这个隐藏列就相当于一个指针,可以通过它来找到该记录修改前的信息。

ReadView

  • 对于使用READ UNCIMMITTED隔离级别的事务来说,由于可以读到未提交事务修改过的记录,所以直接读取记录的最新版本就好了;
  • 对于使用READ COMMITTED 和REPEATABLE READ 隔离级别的事务来说,都必须保证读到已经提交了的事务修改过的记录,也就是说假如另一个事务已经修改了记录但是尚未提交,是不能直接读取到最新版本记录的。核心问题:需要判断一下版本链中的哪个版本是当前事务可见的。为此设计了readview
  • readView包含4个比较重要的内容:
    • m_ids:表示在生成ReadView时当前系统中活跃的读写事务的事务ID
    • min_trx_id:表示生成ReadView时当前系统中活跃的读写事务中最小的事务id,也就是m_ids中的最小值
    • max_trx_id: 表示生成ReadView时系统中应该分配给下一个事务的id值
    • creator_trx_id:表示生成该ReadView的事务的事务ID
      • 我们前边说过,只有在对表中的记录做改动时(执行INSERT、DELETE、UPDATE这些语句时)才会为事务分配事务id,否则在一个只读事务中的事务id值都默认为0。
  • 有了这个ReadView,这样在访问某条记录时,只需要按照下边的步骤判断记录的某个版本是否可见:
    • 如果被访问的版本的trx_id属性与ReadView中的creator_trx_id相同,意味着当前事务在访问它自己修改过的记录,所以该版本可以被当前事务访问
    • 如果被访问的trx_id属性值小于ReadView中的min_trx_id值,表明生成该版本的事务在当前事务生成ReadView时已经提交,所以该版本可以被当前事务访问
    • 如果被访问版本的trx_id属性值大于或等于ReadView中的max_trx_id值,表明生成该版本的事务在当前事务生成ReadView后才开启,所以该版本不可以被当前事务访问
    • 如果被访问版本的trx_id属性值在ReadView的min_trx_id和max_trx_id之间,那就需要判断一下trx_id属性值是不是在m_ids列表中,如果在,说明创建ReadView时生成该版本的事务还是活跃的,该版本不可被访问;如果不在,说明创建ReadView时生成该版本的事务已经被提交,该版本可以被访问。

总结一下:

  • READ COMMITTED隔离级别的事务在每次查询开始时都会生成一个独立的ReadView
  • REPEATABLE READ :在第一次读取数据时生成一个ReadView,也就是说两次SELECT 查询得到的结果是重复的。

MVCC总结: 所谓的MVCC指的就是在使用 READ COMMITTED 和REPEATABLE READ 这两种隔离级别的事务在执行普通的SELECT 操作时访问的记录的版本链的过程,这样子可以使不用的事务的读-写、写-读操作并发执行,从而提升性能。

mysql如何在RR级别解决幻读的

1.当前读,读的是最新版本,并且需要获取对应记录的锁,如下SQL

  • select ... lock in share mode
  • select ... for update
  • update 、delete 、insert

是通过next-key 来实现幻读的

2.快照读 是通过mvcc 来解决的

以上就是详解MySQL事务的隔离级别与MVCC的详细内容,更多关于MySQL事务的隔离级别与MVCC的资料请关注三水点靠木其它相关文章!

MySQL 相关文章推荐
MySQL分库分表与分区的入门指南
Apr 22 MySQL
MySQL表字段时间设置默认值
May 13 MySQL
mysql如何配置白名单访问
Jun 30 MySQL
mysql备份策略的实现(全量备份+增量备份)
Jul 07 MySQL
SQL实现LeetCode(176.第二高薪水)
Aug 04 MySQL
Mysql实现简易版搜索引擎的示例代码
Aug 30 MySQL
MySQL数据库中varchar类型的数字比较大小的方法
Nov 17 MySQL
mysql中整数数据类型tinyint详解
Dec 06 MySQL
MySQL中一条SQL查询语句是如何执行的
Apr 08 MySQL
深入理解MySQL中MVCC与BufferPool缓存机制
May 25 MySQL
MySQL数据库简介与基本操作
May 30 MySQL
MySQL选择合适的备份策略和备份工具
Jun 01 MySQL
MySQL之高可用集群部署及故障切换实现
教你用eclipse连接mysql数据库
MySQL 慢查询日志深入理解
MySQL root密码的重置方法
MySQL性能压力基准测试工具sysbench的使用简介
Apr 21 #MySQL
Mysql MVCC机制原理详解
详解MySQL 用户权限管理
You might like
Discuz Uchome ajaxpost小技巧
2011/01/04 PHP
php开发文档 会员收费1期
2012/08/14 PHP
PHP分页详细讲解(有实例)
2013/10/30 PHP
PHP框架实现WebSocket在线聊天通讯系统
2019/11/21 PHP
js 分页全选或反选标识实现代码
2011/08/09 Javascript
多种方法判断Javascript对象是否存在
2013/09/22 Javascript
node.js中的fs.fstat方法使用说明
2014/12/15 Javascript
vueJS简单的点击显示与隐藏的效果【实现代码】
2016/05/03 Javascript
node.js与C语言 实现遍历文件夹下最大的文件,并输出路径,大小
2017/01/20 Javascript
JS验证全角与半角及相互转化的介绍
2017/05/18 Javascript
AngularJS 最常用的八种功能(基础知识)
2017/06/26 Javascript
微信小程序引用公共js里的方法的实例详解
2017/08/17 Javascript
vue2.0 datepicker使用方法
2018/02/04 Javascript
微信小程序使用gitee进行版本管理
2018/09/20 Javascript
对 Vue-Router 进行单元测试的方法
2018/11/05 Javascript
javascript/jquery实现点击触发事件的方法分析
2019/11/11 jQuery
js调用网络摄像头的方法
2020/12/05 Javascript
[01:22:42]2014 DOTA2华西杯精英邀请赛 5 24 DK VS LGD
2014/05/26 DOTA
[02:37]2018DOTA2亚洲邀请赛赛前采访 VP.no[o]ne心中最强SOLO是谁
2018/04/04 DOTA
简介Python设计模式中的代理模式与模板方法模式编程
2016/02/02 Python
Django-Rest-Framework 权限管理源码浅析(小结)
2018/11/12 Python
对pyqt5多线程正确的开启姿势详解
2019/06/14 Python
python使用 cx_Oracle 模块进行查询操作示例
2019/11/28 Python
关于ZeroMQ 三种模式python3实现方式
2019/12/23 Python
Keras 加载已经训练好的模型进行预测操作
2020/06/17 Python
教你一分钟在win10终端成功安装Pytorch的方法步骤
2021/01/28 Python
N.Peal官网:来自伦敦的高档羊绒品牌
2018/10/29 全球购物
女士和男士时尚鞋在线购物:Shoespie
2019/02/28 全球购物
廉价连衣裙和婚纱礼服在线销售:Tbdress
2019/02/28 全球购物
Rhone官方网站:男士运动服装、健身服装和高级运动服
2019/05/01 全球购物
平安校园建设方案
2014/05/02 职场文书
党员干部廉洁承诺书
2014/05/28 职场文书
大学生工作自荐书
2014/06/16 职场文书
社区志愿者活动总结
2014/06/26 职场文书
大学毕业生自我鉴定范文
2019/06/21 职场文书
浅谈golang 中time.After释放的问题
2021/05/05 Golang