MySQL - Readview
2022-02-22 本文已影响0人
祁小彬
在隔离级别中提到:
- 对于使用READ UNCOMMITTED隔离级别的事务来说,直接读取记录的最新版本就好了。
- 对于使用SERIALIZABLE隔离级别的事务来说,使用加锁的方式来访问记录。
- 对于使用READ COMMITTED和REPEATABLE READ隔离级别的事务来说,就需要通过ReadView追溯版本链。
核心问题就是:需要判断一下版本链中的哪个版本是当前事务可见的。所以设计InnoDB的设计者提出了一个ReadView的概念,这个ReadView中主要包含当前系统中还有哪些活跃的读写事务,把它们的事务id放到一个列表中,我们把这个列表命名为为m_ids。
这样在访问某条记录时,只需要按照下边的步骤判断该记录在版本链中的某个版本(trx_id)是否可见:
- trx_id < m_ids列表中最小的事务id
表明生成该版本的事务在生成ReadView前已经提交,所以该版本可以被当前事务访问。 - trx_id > m_ids列表中最大的事务id
表明生成该版本的事务在生成ReadView 后才生成,所以该版本不可以被当前事务访问。 - m_ids列表中最小的事务id < trx_id < m_ids列表中最大的事务id
(1)trx_id在m_ids中:说明创建 ReadView 时生成该版本的事务还是活跃的,该版本不可以被访问。
(2)trx_id不在m_ids中:说明创建 ReadView 时生成该版本的事务已经被提交,该版本可以被访问。
如果某个版本的数据对当前事务不可见的话,那就顺着版本链找到下一个版本的数据,继续按照上边的步骤判断可见性,依此类推,直到版本链中的最后一个版本,如果最后一个版本也不可见的话,那么就意味着该条记录对该事务不可见,查询结果就不包含该记录。