Mysql中的快照读和当前读

mysql当前读和快照读(MVCC)

mysql读取数据实际上有两种读取模式:当前读和快照读

当前读:每次读取的都是当前最新的数据,但是读的时候不允许写,写的时候也不允许读。

快照读:读写不冲突,每次读取的是快照数据,

隔离级别Repeatable Read下(默认隔离级别):有可能读取的不是最新的数据

Read Committed隔离级别下:快照读和当前读读取的数据是一样的,都是最新的。

一. 相关知识

1.1 undolog

在操作数据之前,把需要操作的数据和事务记录备份到undo log中,目的就是为了保证原子性,如果一个事务执行中发生了故障,就可以通过undolog进行回滚

1.2 共享锁和排它锁

共享锁(S锁):共享 (S) 用于不更改或不更新数据的操作(只读操作),如 SELECT 语句。

如果事务T仅对数据A进行读取,那么会对数据A加上共享锁,之后则其他事务如果要读取数据A的话可以对其继续加共享锁,但是不能加排他锁(也就是无法修改数据)。获准共享锁的事务只能读数据,不能修改数据。

排他锁(X锁):用于数据修改操作,例如 INSERT、UPDATE 或 DELETE。确保不会同时同一资源进行多重更新。

如果事务T对数据A要进行修改,则需要对其添加排它锁,加上排他锁后,则其他事务不能再对A加任任何类型的封锁。获准排他锁的事务既能读数据,又能修改数据。

二. 当前读

Mysql实现当前读是通过共享锁+排他锁+Next-Key Lock实现的。

每次对行数据进行读取的时候,加共享锁。此时就不允许修改,但是允许其他事务读取,所以每次都可以读到最新的数据。

每次对行数据进行修改的时候,加排他锁,不允许其他事务读取和修改。这种情况下其他事务读取的数据也一定是最新的数据。

每次对范围行数据进行读取的时候,对这个范围加一个范围共享锁。

每次对范围行数据进行修改的时候,读这个范围加一个范围排它锁。

基于上述锁机制,实现当前读,确保每次读取的都是最新的数据。

三. 快照读

mysql中的快照读是通过MVCC+undolog实现的。

快照读,顾名思义,就是读取快照数据,也就是说当某个数据正在被修改的时候,也可以进行读取该数据,保证读写不冲突。

刚刚提到undolog,当我们对记录做了变更操作时,就会产生undo记录,undo记录中存储的是老版数据,当一个旧的事务需要读取数据时,为了能够读取到老版本的数据,需要顺着undo列找到满足其可见性的记录,这个找满足可见行的记录依赖。就是说每次都是读取undolog中的数据。

MVCC:

MVCC也称多版本并发控制,说的通俗易懂一点就是记录数据的不同版本。

实际上,不同事务或者相同事务的对同一记录的修改,会导致该记录的undo log成为一条记录版本线性表,既链表,undo log的链首就是最新的旧记录,链尾就是最早的旧记录。

所以快照读都是去读取undolog中链首的最新的旧记录。

但是不同隔离级别下的快照读又有所区别。下面简单阐述

四. 不同隔离级别下的快照读

这里举例说明:

事务开始前 name = 张三 id = 1

事务A:select name from t where id = 1 (多次)

事务B:update name = 李四 from t where id = 1

在Read Committed隔离级别下,快照读和当前读读取的是一样的。因为每次select都会去读取最新的快照数据。

过程如下:

事务A开始

事务B开始

事务A select ,查出name = 张三

事务B update,修改name = 李四

事务B结束

事务A 再次select ,查出name = 李四 (因为此时undolog中的最新数据为李四)

事务A结束

在默认隔离级别Repeatable Read下,在一个事务内,读取的都是第一次select的数据,所以可能会出现读取的数据不是最新数据的情况

过程如下:

事务A开始

事务B开始

事务A select ,查出name = 张三

事务B update,修改name = 李四

事务B结束

事务A 再次select ,查出name = 张三 (因为只读第一次的快照数据)

事务A结束

相关推荐
不羁。。7 小时前
【撸靶笔记】第八关:GET - Blind - Boolian Based - Single Quotes
数据库·sql·mybatis
AwhiteV8 小时前
利用图数据库高效解决 Text2sql 任务中表结构复杂时占用过多大模型上下文的问题
数据库·人工智能·自然语言处理·oracle·大模型·text2sql
m0_595199858 小时前
Redis(以Django为例,含具体操作步骤)
数据库·redis·缓存
爱尚你19938 小时前
MySQL 三大日志:redo log、undo log、binlog 详解
数据库·mysql
小猿姐9 小时前
KubeBlocks AI:AI时代的云原生数据库运维探索
数据库·人工智能·云原生·kubeblocks
NocoBase11 小时前
10 个开源工具,快速构建数据应用
数据库·低代码·开源
麻辣清汤11 小时前
结合BI多维度异常分析(日期-> 商家/渠道->日期(商家/渠道))
数据库·python·sql·finebi
钢铁男儿11 小时前
Python 正则表达式(正则表达式和Python 语言)
python·mysql·正则表达式
Kan先生13 小时前
对象存储解决方案:MinIO 的架构与代码实战
数据库·python
超级迅猛龙13 小时前
保姆级Debezium抽取SQL Server同步kafka
数据库·hadoop·mysql·sqlserver·kafka·linq·cdc