详解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 05 MySQL
MySQL 重命名表的操作方法及注意事项
May 21 MySQL
MySQL中的布尔值,怎么存储false或true
Jun 04 MySQL
MySQL 常见的数据表设计误区汇总
Jun 07 MySQL
MySQL系列之二 多实例配置
Jul 02 MySQL
MySQL 用 limit 为什么会影响性能
Sep 15 MySQL
浅谈MySQL表空间回收的正确姿势
Oct 05 MySQL
Mysql关于数据库是否应该使用外键约束详解说明
Oct 24 MySQL
mysql 联合索引生效的条件及索引失效的条件
Nov 20 MySQL
MySQL中rank() over、dense_rank() over、row_number() over用法介绍
Mar 23 MySQL
Windows 64位 安装 mysql 8.0.28 图文教程
Apr 19 MySQL
Mysql中常用的join连接方式
May 11 MySQL
MySQL之高可用集群部署及故障切换实现
教你用eclipse连接mysql数据库
MySQL 慢查询日志深入理解
MySQL root密码的重置方法
MySQL性能压力基准测试工具sysbench的使用简介
Apr 21 #MySQL
Mysql MVCC机制原理详解
详解MySQL 用户权限管理
You might like
php float不四舍五入截取浮点型字符串方法总结
2013/10/28 PHP
PHP+jquery+ajax实现即时聊天功能实例
2014/12/23 PHP
WordPress网站性能优化指南
2015/11/18 PHP
PHP模板引擎Smarty内建函数详解
2016/04/11 PHP
PHP检查URL包含特定字符串实例方法
2019/02/11 PHP
PHP的curl函数的用法总结
2019/02/14 PHP
laravel 出现command not found问题的解决方案
2019/10/23 PHP
使用EXT实现无刷新动态调用股票信息
2008/11/01 Javascript
Javascript查询DBpedia小应用实例学习
2013/03/07 Javascript
浅析showModalDialog数据缓存问题(用禁止浏览器缓存解决)
2013/07/09 Javascript
在css加载完毕后自动判断页面是否加入css或js文件
2014/09/10 Javascript
JavaScript实现在页面间传值的方法
2015/04/07 Javascript
javascript清空table表格的方法
2015/05/14 Javascript
JS判断页面是否出现滚动条的方法
2015/07/17 Javascript
详解JavaScript中的Unescape()和String() 函数
2015/11/09 Javascript
浅谈JS正则表达式的RegExp对象和括号的使用
2016/07/28 Javascript
iscroll.js滚动加载实例详解
2017/07/18 Javascript
javascript原型链学习记录之继承实现方式分析
2019/05/01 Javascript
Vue中jsx不完全应用指南小结
2019/11/01 Javascript
JavaScript计算正方形面积
2019/11/26 Javascript
jQuery实现可以计算进制转换的计算器
2020/10/19 jQuery
HTML元素拖拽功能实现的完整实例
2020/12/04 Javascript
解析Python编程中的包结构
2015/10/25 Python
python微信好友数据分析详解
2018/11/19 Python
Python WEB应用部署的实现方法
2019/01/02 Python
python字典排序的方法
2019/10/12 Python
python访问hdfs的操作
2020/06/06 Python
使用python脚本自动生成K8S-YAML的方法示例
2020/07/12 Python
Python远程linux执行命令实现
2020/11/11 Python
Puma印度官网:德国运动品牌
2019/10/06 全球购物
服装设计行业个人的自我评价
2013/12/20 职场文书
计算机科学系职业生涯规划书
2014/03/08 职场文书
个人投资计划书
2014/05/01 职场文书
报考公务员诚信承诺书
2014/08/29 职场文书
学校教师师德师风承诺书
2015/04/28 职场文书
趣味运动会加油词
2015/07/18 职场文书