mysql面试题——MVCC

本文介绍了MVCC(多版本并发控制)的概念,如何处理读写冲突,以及其依赖的隐藏字段、UndoLog和ReadView。详细解释了快照读与当前读的区别,以及MVCC的实现原理和操作流程。以不同隔离级别为例,展示了MVCC在实际应用中的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一:什么是MVCC?

多版本并发控制,更好的方式去处理读-写冲突,就是为了查询一些正在被另一个事务更新的行,并且可以看到它们被更新之前的值,这样在做查询的时候就不用等待另一个事务释放锁。

二: 快照读与当前读

快照图:读取的是快照数据,不加锁的简单的 SELECT 都属于快照读

SELECT * FROM player WHERE ...

当前读:读取的是记录的最新版本,加锁的 SELECT,或者对数据进行增删改都会进行当前读

SELECT * FROM XX_table LOCK IN SHARE MODE;
SELECT * FROM xx_table FOR UPDATE;
INSERT INTO xx_table ...
DELETE FROM xx_table ...
UPDATE xx_table ...

三:MVCC实现原理

MVCC 的实现依赖于:隐藏字段、Undo Log、Read View。

  1. 隐藏字段:trx_id,roll_pointer在这里插入图片描述
  2. 在这里插入图片描述
  3. ReadView:Readview 就是事务在使用MVCC机制进行快照读操作时产生的读视图。当事务启动时,会生成数据库系统当前的一个快照,InnoDB 为每个事务构造了一个数组,用来记录并维护系统当前活跃事务的ID(“活跃”指的就是,启动了但还没提交)。

四:MVCC整体操作流程

  1. 首先获取事务自己的版本号,也就是事务 ID;
  2. 获取 ReadView;
  3. 查询得到的数据,然后与 ReadView 中的事务版本号进行比较;
  4. 如果不符合 ReadView 规则,就需要从 Undo Log 中获取历史快照;
  5. 最后返回符合规则的数据。
    ReadView的规则
    (1)如果被访问版本的trx_id属性值与ReadView中的 creator_trx_id 值相同,意味着当前事务在访问它自己修改过的记录,所以该版本可以被当前事务访问。
    (2)如果被访问版本的trx_id属性值小于ReadView中的 up_limit_id 值,表明生成该版本的事务在当前事务生成ReadView前已经提交,所以该版本可以被当前事务访问。
    (3)如果被访问版本的trx_id属性值大于或等于ReadView中的 low_limit_id 值,表明生成该版本的事务在当前事务生成ReadView后才开启,所以该版本不可以被当前事务访问。
    (4)如果被访问版本的trx_id属性值在ReadView的 up_limit_id 和 low_limit_id 之间,那就需要判断一下trx_id属性值是不是在 trx_ids 列表中。
    如果在,说明创建ReadView时生成该版本的事务还是活跃的,该版本不可以被访问。
    如果不在,说明创建ReadView时生成该版本的事务已经被提交,该版本可以被访问

五:举例说明

读已提交(READ COMMITTED)隔离级别下:一个事务中的每一次 SELECT 查询都会重新获取一次Read View。
在这里插入图片描述

可重复读隔离级别下:因为一个事务只在第一次 SELECT 的时候会获取一次 Read View,而后面所有的 SELECT 都会复用这个 Read View
在这里插入图片描述

### MySQL MVCC 面试题及答案 #### 什么是多版本并发控制 (MVCC),其主要优点是什么? MVCC 是一种用于数据库管理系统的技术,允许多个事务同时访问同一数据的不同版本而不互相干扰。最大的好处在于读操作不需要加锁,读写之间也不会发生冲突,这大大提高了 MySQL 数据库系统的并发性能[^1]。 #### InnoDB 中如何实现 MVCC 来保证事务的隔离性? InnoDB 使用隐藏列 `DB_TRX_ID` 记录最近更新这条记录的事物 ID 和 `DB_ROLL_PTR` 指向回滚段的位置。每次修改数据时,旧的数据会被保存到 undo 日志中形成一个链表结构。当有新的查询请求到来时,系统会根据当前事物的状态创建 ReadView 并沿着版本链寻找符合条件的第一个版本返回给用户[^5]。 #### 不同隔离级别下 MVCC 表现有何差异? - **读未提交(READ UNCOMMITTED)**: 这种情况下实际上并没有真正利用上 MVCC 特性,因为可以看到其他尚未提交事务所做的更改。 - **读已提交(READ COMMITTED)**: 每次执行 SELECT 查询都会重新计算一个新的 ReadView ,这意味着即使在同一事务内也可能看到不同的结果集[^3]。 - **可重复读(REPEATABLE READ)**: 整个交易期间只会有一个固定的 ReadView 。因此,在同一个事务内的多次相同条件下的 select 操作总是能看到一致的结果集合。 - **串行化(SERIALIZABLE)**: 此模式强制所有的读取都采用共享锁的方式来进行,从而完全避免了幻影读现象的发生。 #### 解析一条 SQL 执行过程中涉及哪些组件以及它们之间的交互流程? 对于每条进入 MySQL Server 的 SQL 请求而言: 1. 客户端通过连接器与服务器建立 TCP/IP 或者 Unix Socket 形式的网络链接; 2. 如果启用了 Query Cache 功能,则在此处检查是否有匹配项可以直接返回; 3. 经过解析器完成词法和语法层面的校验之后生成抽象语法树 AST; 4. 接下来由预处理模块做更深层次的安全性和合法性检测; 5. 优化器依据多种因素评估不同可能路径的成本开销并挑选最优方案作为最终决定; 6. 最终交由执行器按照既定策略调用底层存储引擎 API 获取所需资源直至全部结束为止; 7. 结果逐步反馈至前端应用层面上显示出来的同时也会被缓存起来供后续重用[^4]。 ```sql SELECT * FROM orders WHERE order_date >= '2023-01-01'; ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值