理解 MVCC
MongoDB、MySQL、Oracle、PostgreSQL 等事务型数据库都有 mvcc 的概念。
MVCC: 即多版本并发控制,主要是为了提高数据库的读写性能,让数据库在读写的时候不用去加锁。mvcc 主要是处理读请求,这个读指的是快照读,而不是当前读,快照读就是普通的 select 查询。而当前读,其实就是一种悲观锁,要靠加锁实现,比如我们执行 update 或 delete 的时候,你需要先把数据读出来,然后再进行修改操作。比如 select xx for update, 这种排他锁,还有 select xxx lock in share mode 共享锁,这种加锁的方式就是当前读,是需要用锁来保证数据的强一致性。快照读是用 mvcc 实现的,他的目的是读写数据行的时候,不用去竞争锁,提升数据库的并发性能。
数据库的事务有 acid 的特性,原子性用 undo log 实现,持久性是 redo log,而隔离性则是通过加锁和 mvcc 实现。 mysql 事务有 4 中隔离级别: 读未提交,读已提交,可重复读,串行化。一般常用的是读已提交,可重复读这两种。读已提交,可重复读 他们的快照读都是基于 mvcc 实现的。
mvcc 常见的几个概念:(图片来源百度)
(1)undo log
(2)版本链
id = 1 这条记录,经过了多次修改。 修改的历史版本都保存在了 undo log 里。数据库里的每行记录实际上包含一个回滚指针 ( 这是一个隐藏字段 ),指向上一个版本记录,如果你想得到 name=”张三“ 的记录,需要通过版本链多次回滚才行。
(3)Readview 可见性
readview 其实就是一次快照(相当于python 的 一个 class 对象),但是有个问题,当查询 select id=1 的时候,我应该读哪个版本呢,难道总是读最新吗? readview 的作用就是让你知道,你要选择读哪个快照版本。
readview 有一些参数:
m_ids: 表示在生成 Readview 时当前系统中活跃(指还未commit 的事务)的读写事务的 事务id 列表。
min_trx_id: 表示在生成 readview 时当前系统中活跃的读写事务最小的 事务id,也就是 m_ids 中最小的值。
max_trx_id: 表示生成 readview 时系统中应该分配下一个事务的 id 值。
create_trx_id: 表示生成 readview 的事务的 事务id。也就是谁生成了这个 readview。(我自己的事务 id。相当于是 自己记录自己。 )
可能不太直观,举个栗子如下:
eg:当前实例中 show proceeelist 查看活跃的事务列表是 m_ids = [2,3,4, 5] ,那么 min_trx_id=2,max_trx_id=6。
readview 视图如何判断版本链中的哪个版本可用呢?他有4 种判断逻辑
【1】trx_id == create_trx_id: 可以访问这个版本; # trx_id 指的就是如下图中的每条记录的事务 ID。 trx_id == create_trx_id 意味着,这条记录是我自己本身(当前 session)生成的事务(readview),我读取我自己创建的这条记录,当然是可读的。别人不可读是为了阻塞,我自己不可能阻塞我自己,没有意义。
【2】trx_id < min_trx_id:可以访问这个版本。说明,我这个 trx_id 就不是一个活跃的 id ,他已经commit 了 。
【3】trx_id > max_trx_id: 不可以访问这个版本。因为 该 readview 还没有生成。
【4】min_trx_id <= trx_id <= max_trx_id : 如果 trx_id 在 m_ids 中,那么是不可以访问这个版本的,在 m_ids 列表里 说明当前事务是活跃事务。
MVCC 是针对 RC 和 RR 隔离级别的:
RC 和 RR 级别 本质的区别是他俩 生成 readview 的时机不同:
【1】RC 中 以每个 select 的查询为单位(update, delete 的查询也算),每次 select 都会生成一个新的 readview,如果一个事务执行的过程中执行了多次 select,那么事务执行的过程中就要生成多个 readview 。本质是一个事务生成了多了 readview 。 RC 有不可重复读的问题。
【2】RR 生成 readview 是以事务为单位的。比如一个事务有三个 select 语句,但是只会生成一个 readview,后面两个 select 使用的 readview 和 第一个一样,他不会再继续生成 readview,所以在可重复读级别下, readview 是以事务为级别的 ,一个事务只会生成一个 readview。RR 级别就是用这个方法解决了不可重复读的问题。
Eg: 我现在开启了一个事务,包含了两个 select 动作,在执行第一个 select 的时候生成了这个 readview,第一个 select 查询到的是 name=“张三”,那么就算另一个 session 的 update 事务 commit 了,我这个事务的第二个 select 查询用到的依然是 第一个 select 的 readview name = “张三”,第一次生成的 readview 是什么样,之后的这个 readview 就是什么样,直到我这个事务执行完毕。 RR 虽然解决了不可重复的问题,但还是存在 幻读的问题,所以要加锁( 间隙锁 + next key lock )解决幻读的问题。 幻读只是在 RR 级别才会出现的。
我们知道 快照读是通过 mvcc 实现的,当前读都是通过加锁实现的:快照读是如何解决幻读的问题呢?
RR 级别因为事务开始的时候只会生成一分 readview,那么外界再怎么对这个数据进行增加,都对我这个readview 视图没有影响,我读的还是我第一次生成的视图,外部的数据再怎么更新,新增,对于我这个视图都没有影响,这种方式解决了幻读的问题。
当前读是通过 间隙锁 + next key lock 实现,间隙锁是锁住了一段范围。RR 级别默认开启了间隙锁。
理解 MVCC的更多相关文章
- 【Mysql】深入理解 MVCC 多版本并发控制
MVCC MVCC(Multi-Version Concurrency Control),即多版本并发控制.是 innodb 实现事务并发与回滚的重要功能.锁机制可以控制并发操作,但是其系统开销较大, ...
- 多版本号并发控制(MVCC)在分布式系统中的应用
QQ群:289150599 问题 近期项目中遇到了一个分布式系统的并发控制问题.该问题能够抽象为:某分布式系统由一个数据中心D和若干业务处理中心L1,L2 ... Ln组成:D本质上是一个key-va ...
- InnoDB MVCC RR隔离级别下的数据可见性总结
一.背景 熟悉数据库隔离级别的人都知道,在RR(可重复读)隔离级别下,无论何时多次执行相同的SELECT快照读语句,得到的结果集都是完全一样的,即便两次SELECT语句执行期间,其他事务已经改变了该查 ...
- HBase并行写机制(mvcc)
HBase在保证高性能的同时,为用户提供了便于理解的一致性数据模型MVCC (Multiversion Concurrency Control),即多版本并发控制技术,把数据库的行锁与行的多个版本结合 ...
- 转: 多版本并发控制(MVCC)在分布式系统中的应用 (from coolshell)
from: http://coolshell.cn/articles/6790.html 问题 最近项目中遇到了一个分布式系统的并发控制问题.该问题可以抽象为:某分布式系统由一个数据中心D和若干业务 ...
- 多版本号并发控制(MVCC)在实际项目中的应用
近期项目中遇到了一个分布式系统的并发控制问题.该问题能够抽象为:某分布式系统由一个数据中心D和若干业务处理中心L1,L2 - Ln组成:D本质上是一个key-value存储,它对外提供基于HTTP协议 ...
- Kudu的概念术语
不多说,直接上干货! Columnar Data Store(列式数据存储) Kudu 是一个 columnar data store(列式数据存储).列式数据存储在强类型列中.由于几个原因,通过适当 ...
- 极客时间 mysql实战45讲下载读 08讲事务到底是隔离的还是不隔离的 笔记
笔记体会: 1.innodb支持RC和RR隔离级别实现是用的一致性视图(consistent read view) 2.事务在启动时会拍一个快照,这个快照是基于整个库的.基于整个库的意思就是说一个事务 ...
- MySQL实战45讲学习笔记:第八讲
一.今日内容概要 我在第 3 篇文章和你讲事务隔离级别的时候提到过,如果是可重复读隔离级别,事务 T 启动的时候会创建一个视图 read-view,之后事务 T 执行期间,即使有其他事务修改了数据,事 ...
随机推荐
- 第一个OC类
1.如何声明一个类 格式 注意: 1.必须以@interface开头,@end结尾 2.成员变量的声明,必须写在@interface与@end之间的大括号中 3.方法的声明必须在{}下面,不能写在{} ...
- JavaEE复制后项目出错或者无法运行的解决方法
eclipse中,直接复制项目,会出现一些错误,解决方法如下: 1,在复制后的工程上点右键->properties->Web Project Settings>web context ...
- Solution -「HDU 6643」Ridiculous Netizens
\(\mathcal{Description}\) Link. 给定一棵含有 \(n\) 个结点的树,点 \(u\) 有点权 \(w_u\),求树上非空连通块的数量,使得连通块内点权积 \(\ ...
- MyBatis中使用log4j进行调试入门实例
导入log4j.jar 设置日志级别等相关内容 文件内容(仅控制台有效): ### 设置###log4j.rootLogger = debug,stdout,D,E### 输出sql信息到控制抬 ## ...
- VS2019配置eigen
本文讲述如何在VS2019中配置eigen eigen版本:eigen-3.3.9 百度网盘地址:https://pan.baidu.com/s/1Bu5A58qV2n8doDs4NpPfJQ 提取 ...
- close-on-exec 相关的一个 bug
close-on-exec 相关的一个 bug 测试一个用 V4L2 拍照的程序时,发现程序单独运行很正常,但在多进程环境下运行时就会出现问题,具体表现为执行 open 系统调用打开 /dev/vid ...
- FastDFS安装和简介详细总结
1.fastDFS简介 1 FastDFS是用c语言编写的一款开源的分布式文件系统. 2 FastDFS为互联网量身定制,充分考虑了冗余备份.负载均衡.线性扩容等机制,并注重高可用.高性能等指标, 3 ...
- 【C#反射】BindingFlags 枚举
BindingFlags 枚举用途:Type的类方法中,用于筛选成员. type.InvokeMember方法中 type.GetConstructor 方法中 type.GetFiles方法中 ty ...
- 基于IEC61499标准的组件
IEC 61499标准通常用于开发分布式控制系统.IEC 61499定义了功能模块(FB, function block)的模型.有三种FB,分别是Basic FB (BFB),Service Int ...
- ssh静态代理模式讲解与使用
--作者:飞翔的小胖猪 --创建时间:2021年5月18日 --修改时间:2021年5月23日 一.说明 SSH 为建立在应用层基础上的安全协议.SSH 是较可靠专为远程登录会话和其他网络服务提供安全 ...