【MySQL】十一、MySQL的RR隔离级别是如何基于ReadView机制实现的
本文主要介绍MySQL数据库的RR隔离级别是如何基于ReadView机制实现的
RR级别下,这个事务读取一条数据,无论读多少次,都是一个值,别的事务修改数据之后,哪怕提交了,你也是看不到人家修改的值的,这就避免了不可重复读的问题。
同时如果别的事务插入了一些新的数据,你也是读不到的,这样就可以避免幻读问题。这到底是如何实现的呢?
首先我们还是假设有一条数据是事务id=5一个事务插入的,同时此时有事务A和事务B同时在运行,事务A的id是60,事务B的id是70,如下图所示:
这个时候,事务A发起了一个查询,他就是第一次查询就会生成一个ReadView,此时ReadView里的creator_trx_id是60,min_trx_id是60,max_trx_id是71,m_ids是[60, 70],此时ReadView如下图所示:
这个时候事务A基于这个ReadView去查询这条数据,会发现这条数据的trx_id为50,是小于ReadView里的min_trx_id的,说明他发起查询之前,早就有事务插入这条数据还提交了,所以此时可以查到这条原始值得,如下图:
接着就是事务B此时更新了这条数据的值为值B,此时会修改trx_id为70,同时生成一个undo log,而且关键是事务B此时还提交了,也就是说此时事务B已经结束了,如下图所示:
此时ReadView中的m_ids还会是60和70吗?当然是的,因为ReadView一旦生成了就不会改变了,这个时候虽然事务B已经结束了,但是事务A的ReadView里,还是会有60和70两个事务id。他的意思就是在事务A开启查询的时候,事务B当时是在运行的。
接着此时事务A去查询这条数据的值,他会发现此时数据的trx_id是70了,70一方面是在ReadView的min_trx_id和max_trx_id的范围区间的,同时还在m_ids列表中,这说明起码是事务A开启查询的时候,id为70的这个事务B还是在运行的,然后由这个事务B更新了这条数据,所以此时事务A是不能查询到事务B更新的这个值得,因此这个时候继续顺着指针往历史版本链条上去找,如下图:
接着事务A顺着指针找到下面一条数据,trx_id为50,是小于ReadView的min_trx_id的,说明在他开启查询之前,就已经提交了这个事务了,所以事务A是可以查询到这个值的,此时事务A查到的是原始值,如下图:
这样就避免了不可重复读的问题,事务A多次读同一个数据,每次读到的都是一样的值,除非他自己修改了值,否则读到的一直会一样的值,不管别的事务如何修改数据,事务A的ReadView始终是不变的,他基于这个ReadView始终看到的值是一样的。
接下来看看如何解决幻读的问题的,假设现在事务A先用select * from x where id > 10来查询,此时可能查到的就是一条数据,而且读到的是这条数据原始值的那个版本,现在有一个事务C插入了一条数据,然后提交了,如下图所示:
接着此时事务A再次查询,此时会发现符合条件的有2条数据,一条是原始值的那个数据,一条是事务C插入的那条,但是事务C插入的那条数据的trx_id是80,这个80是大于自己ReadView的max_trx_id的,说明是自己发起查询之后,这个事务才启动的,所以此时这条数据是不能查询的。因此事务A本次查询,还是只能查到原始值一条数据。
所以事务A根本不会发生幻读,他根据条件范围查询的时候,每次读到的数据都是一样的,不会读到人家插入进去的数据。
【MySQL】十一、MySQL的RR隔离级别是如何基于ReadView机制实现的