https://www.bilibili.com/video/BV1Kr4y1i7ru/?p=141
https://blog.csdn.net/weixin_52574640/article/details/129961415
MVCC,全称Multo-Version Concurrentcy Control,多版本并发控制。指维护一个数据的多个版本,使得读写操作没有冲突,快照读为MySQL实现MVCC提供了一个非阻塞读功能。MVCC的具体实现,还需要依赖与数据记录中的是哪个隐式字段、undo log日志、readView
当前读:读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。对于我们日常的操作,如:select ... lock in share mode
(共享锁),select … for update 、update、insert、delete(排他锁)都是一种当前读
快照读:简单的select(不加锁)就是快照读,快照读读到的是记录数据的可见版本,有可能是历史数据,不加锁,是非阻塞读。它主要是为了实现可重复读的事务隔离级别
- Read Committed:每次select,都生产一个快照读
- Repatable Read:开启事务后第一个select语句才是快照读的地方。
- Serializable:快照读会退化为当前读。
隐藏字段
每张表除了显式的字段,还有一些隐藏字段
undo log
undo log,回滚日志,在insert、update、delete的时候产生的便于数据回滚的日志。
- 当insert的时候,产生的undo log 日志只在回滚时需要,在事务提交后,可被立即删除
- 而update、delete的时候,产生的undo log 日志不仅在回滚时需要,在快照时也需要,不会立即被删除
undo log版本链
不同事务或相同事务对同一条记录进行修改,会导致该记录的undolog生成一条记录版本链表,链表的头部是最新的旧记录,链表的尾部是最早的旧记录
ReadView
ReadView(读视图)是快照读SQL执行时MVCC提供数据的依据,记录并维护系统当前活跃的事务id。(上面undo log链中,select查询出来的数据是什么样的,由readView决定)
ReadView中包含了四个核心字段:
不同的隔离级别,生成ReadView时机不同:
- Read COMMITTED:在事务中每一次执行快照时生成ReadView
- Repeatable READ:仅在事务中第一次执行快照读时生成ReadView,后续复用该ReadView
依次带入undo log中的DB_TRX_ID,找到符合的,则对应的记录为查询的结果。
事务5第一次查询的结果为0x00002,第二次查询结果为0x00003。(很好理解,RC,读已提交的)