一、MVCC解决了什么问题?
首先我们要知道MySQL的事务隔离级别:
- 读未提交(RU):A事务读取了B事务未提交的数据,可能会引起脏读。
- 读已提交(RC):A事务读取B事务已经提交的数据,可能会引起不可重复读。
- 可重复读(RR):一个事务多次读取同一行数据,结果是相同的,可能会引起幻读。
- 串行化:解决了所有问题,但是开销比较大。
MVCC解决了读已提交中的问题不可重复读。
二、MVCC是什么?
首先我们要知道MVCC:就是多版本(一行数据的版本链)并发控制(读取的结果)。
什么是版本链?
版本链存储的除了我们实际的数据,其实还存储了trx_id(事务ID)和指向上一个版本的指针。
undo log就是用来存储历史数据的。
什么是read view?
read view里面有四个核心参数:
字段 | 含义 |
---|---|
m_ids | 当前活跃的事务ID集合 |
min_trx_id | 最小活跃事务ID |
max_trx_id | 预分配事务ID,当前最大事务ID + 1(因为事务ID是自增的) |
creator_trx_id | ReadView创建者的事务ID |
read view就规定了版本链数据的访问规则。
条件 | 是否可以访问 |
---|---|
trx_id<min_trs_id | 可以访问 |
trx_id=creator_trx_id | 可以访问 |
trx_id>max_trx_id | 不能访问 |
min_trs_id<trx_id<max_trx_id且trx_id在m_ids中 | 不能访问 |
不同的隔离级别生成read view的时机不同。在读已提交中,每一次执行都会生成一个新的read view。在可重复读中,只会在开始执行的时候生成read view,之后的所有操作都使用的是同一个read view。