【数据库】MySQL的ReadView

MySQL的ReadView

前言

1、根据事务的隔离级别,我们已经知道读未提交、读已提交、可重复读、串行化,随着隔离级别的加强,能解决脏写、脏读、不可重复读、幻读的问题。

2、InnoDB 是 MySQL(mysql-5.1版本后) 默认的存储引擎,InnoDB 默认的隔离级别就是可重复读。在这个隔离级别下,开启事务之后,多次读写同一行数据,读到的值永远是一致。

3、当 MySQL 执行写操作之前,会把即将被修改的数据记录到 undo log 日志里面。MySQL可从 undo log 日志中,读取到原插入、修改、删除之前的值,最终把值重新变回去,这就是回滚操作。

undo log 版本链

1、undo log 版本链是基于 undo log 实现的。undo log 中主要保存了数据的基本信息,比如说日志开始的位置、结束的位置,主键的长度、表id,日志编号、日志类型
在这里插入图片描述

此外,undo log 还包含两个隐藏字段 trx_id 和 roll_pointer。trx_id 表示当前这个事务的 id,MySQL 会为每个事务分配一个 id,这个 id 是递增的。roll_pointer 是一个指针,指向这个事务之前的 undo log。
在这里插入图片描述

2、例子
执行:

INSERT INTO student VALUES (1, '张三');

产生:
在这里插入图片描述

继续执行:

UPDATE student SET name='李四' WHERE id=1;

产生:
在这里插入图片描述

继续执行:

UPDATE student SET name='王五' WHERE id=1;

产生:
在这里插入图片描述

为了保证事务并发操作时,在写各自的undo log时不产生冲突,InnoDB采用回滚段的方式来维护undo log的并发写入和持久化。回滚段实际上是一种Undo文件组织方式。

ReadView 机制

ReadView 其实就是一个保存事务ID的list列表。记录的是本事务执行时,MySQL还有哪些事务在执行,且还没有提交。(当前系统中还有哪些活跃的读写事务)

它主要包含这样几部分:

  • m_ids,当前有哪些事务正在执行,且还没有提交,这些事务的 id 就会存在这里;
  • min_trx_id,是指 m_ids 里最小的值;
  • max_trx_id,是指下一个要生成的事务 id。下一个要生成的事务 id 肯定比现在所有事务的 id 都大;
  • creator_trx_id,每开启一个事务都会生成一个 ReadView,而 creator_trx_id 就是这个开启的事务的 id。

这样在访问某条记录时,只需要按照下边的步骤判断该记录在版本链中的某个版本(trx_id)是否可见:
1、trx_id < m_ids列表中最小的事务id
表明生成该版本的事务在生成ReadView前已经提交,所以该版本可以被当前事务访问。
2、trx_id > m_ids列表中最大的事务id
表明生成该版本的事务在生成ReadView 后才生成,所以该版本不可以被当前事务访问。
3、m_ids列表中最小的事务id < trx_id < m_ids列表中最大的事务id
此处比如m_ids为[5,6,7,9,10]
①、若trx_id在m_ids中,比如是6,说明创建 ReadView 时生成该版本的事务还是活跃的,该版本不可以被访问。
②、若trx_id不在m_ids中,比如是8:说明创建 ReadView 时生成该版本的事务已经被提交,该版本可以被访问。

一句话说:当trx_id在m_ids中,或者大于m_ids列表中最大的事务id的时候,这个版本就不能被访问。

如果某个版本的数据对当前事务不可见的话,那就顺着版本链找到下一个版本的数据,继续按照上边的步骤判断可见性,依此类推,直到版本链中的最后一个版本,如果最后一个版本也不可见的话,那么就意味着该条记录对该事务不可见,查询结果就不包含该记录。

例子:

事务是可以并发执行的,现在有事务 A、事务 B 这两个事务,且这两个都没有提交。事务 A 将会执行多次读操作,来模拟可重复读中多次读取同一行数据的场景。事务 B 则会修改这一行数据。
在这里插入图片描述

事务 A 开启事务的时候会生成一个 ReadView,所以说这个 ReadView 的创建者就是事务 A,事务 A 的事务 id 是 10,所以 creator_trx_id 就是 10。

此时,总共就只有事务 A、事务 B 这两个事务,而且它们都还没有提交,所以说 m_ids 会把这两个事务 id,10、18 都记录下来。min_trx_id 是 m_ids 里面的最小值,10、18 中最小的显然是 10。当前最大的事务 id 是 18,那么下一个事务的 id 就是 19,max_trx_id 就是 19。

ReadView 生成之后,事务 A 就要去 undo log 版本链中读取值了。

现在只有一条 undo log 日志,但这并不意味着事务 A 就能读到这条日志的值 X。它要先判断这行日志的 trx_id 是否小于当前事务的 min_trx_id。看图我们可以很轻松地发现,日志的 trx_id = 8 小于 ReadView 中 min_trx_id = 10。

这就意味着,这个事务 A 开始执行之前,修改这行数据的事务已经提交了,所以事务 A 是可以查到值 X 的。

在此基础上再增添一点操作,实现可重复读

我们继续看,事务 A 第一次读完之后,事务 B 要修改这行数据了。undo log 会为所有写操作生成日志,所以就会生成一条 undo log 日志,并且它的 roll_pointer 会指向上一条 undo log 日志。

在这里插入图片描述

紧接着,事务 A 第二次去读这行数据了,情况如下图所示:
在这里插入图片描述
第一次读的时候,开启事务 A 的时候就生成了一个 ReadView

此时事务 A 第二次去查询的时候,先查到的是 trx_id = 18 的那条数据,它会发现 18 比最小的事务编号 10 大。那就说明事务编号为 18 的事务,有可能它是读不到的。

接着就要去 m_ids 里确认是否有 18 这条数据了。发现有 18,那就说明在事务 A 开启事务的时候,这个事务是没有提交的,它修改的数据就不应该被读到。

事务 A 就会顺着 roll_pointer 指针继续往下找,找到了 trx_id = 8 这条日志,发现这条能读,读到的值任然是 x,与第一次读到的结果一致。实现可重复读。

  • 55
    点赞
  • 64
    收藏
    觉得还不错? 一键收藏
  • 18
    评论
### 回答1: MySQLRead view是InnoDB存储引擎实现MVCC(多版本并发控制)的一部分。它主要用于支持在并发读取事务数据时保证数据的一致性。在MVCC中,每个事务都可以看到数据库中的一个快照版本。在并发读取时,Read view可以确保每个事务看到的数据版本是相同的,从而确保数据的一致性。 Read view是一个由系统版本号和活动事务列表组成的数据结构。系统版本号表示当前数据库中的最新版本,而活动事务列表则包含正在运行的事务和已提交但尚未释放其快照版本的事务。在读取数据时,MySQL会使用当前的Read view来确定事务所能看到的数据版本,并且只返回在此版本之前提交的数据。 因此,Read viewMySQL中起着非常重要的作用,能够保证数据的一致性和并发性能。 ### 回答2: MySQLRead view(读视图)是一种用于查询事务中数据的逻辑快照。它的作用主要有以下几点。 首先,Read view用于解决并发事务中的读写冲突。当一个事务开始时,它会创建一个自己的Read view,该视图包含了当前事务开始时数据库中数据的一个快照。其他并发事务可以继续对数据库进行增删改操作,但读取该数据的事务只能读取到Read view中的数据。这样可以保证读取事务在并发环境下不会读到其他未提交的事务的脏数据,确保读操作的一致性和准确性。 其次,Read view用于实现事务的隔离级别。MySQL支持多种事务隔离级别,包括读未提交、读提交、可重复读和串行化。每种隔离级别在Read view中有不同的实现方式,从而保证不同隔离级别下的事务读取到的数据是符合要求的。 另外,Read view还可以用于MVCC(多版本并发控制)机制。MVCC在数据库中可以实现事务的并发控制,通过为每个事务分配唯一的事务标识和Read view来实现。Read view中记录了每个事务开始时数据库中数据的一个快照,可以对比该快照和数据库当前的版本,判断其他事务的读写冲突,进而进行相应的并发控制,提高数据库的并发性能。 总之,MySQLRead view在并发事务中起到了保证读操作的一致性和隔离性的作用,同时也是实现MVCC机制的重要组成部分。通过使用Read view,可以使得事务在读取数据时不会受到其他并发事务的干扰,并保持数据库的一致性。 ### 回答3: MySQLRead view是指在并发读取过程中,为每个事务提供一致的读取快照。它的作用主要体现在以下几个方面: 首先,Read view可以解决读取过程中的脏读问题。当一个事务在执行过程中,其他事务可能会对相同的数据进行修改,如果没有Read view,就可能导致读取到未提交的脏数据。而通过使用Read view,每个事务都能读取到其他事务已提交的数据,避免了脏读问题的发生。 其次,Read view可以提供一致的读取视图。当一个事务开始时,Read view会记录下事务开始时数据库中所有数据的版本号。事务执行过程中,如果其他事务对数据进行了修改并提交了,那么这个事务仍然看到的是开始时的数据版本,保证了事务内部的一致性。 此外,Read view还可以提高并发读取的效率。在MySQL中,Read view通过MVCC(多版本并发控制)来实现。每个事务根据开始时的版本号,只读取那些已提交版本中的数据,可以避免了锁机制的使用,从而提高了并发读取的效率。 总的来说,MySQLRead view提供了一致的读取视图,解决了并发读取过程中可能出现的脏读问题,同时还提高了并发读取的效率。它是保证数据的一致性和并发性的重要机制之一。
评论 18
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值