幻读指的是,在同一个事务中,以同样的条件执行的两次查询,第二次查询查到了第一次查询所没查到的数据。
在mysql的四种隔离级别中,可重复读和串行化两种隔离级别没有幻读问题。那么它们是如何解决幻读问题的呢?
先说串行化这个隔离级别,串行化是通过加锁的方式,让事务按照顺序串行执行,事务会对整张表加锁,读操作时加共享锁,写操作时加排它锁,总之,事务会对整张表加锁,并且这个锁要到事务结束时才释放,所以串行化这个隔离级别根本就不会有幻读问题。
再说可重复读,在可重复读这个隔离级别下,快照读是不会有幻读问题的【也可以说快照读用MVCC避免了幻读问题】,只有当前读会有幻读问题,那么什么是快照读和当前读呢?
快照读读的是快照数据,是由MVCC机制实现的,因为读的是快照数据,所以即便是有其他事务并发修改了数据或者添加了新的数据,也不会对当前事务的快照读造成影响,因为当前事务的快照读读的是同一个版本的快照,所以快照读不会有幻读问题。读已提交 和 可重复读这两种隔离级别下的所有的普通的select查询都是快照读,除了快照读之外就都是当前读了。
当前读读的永远都是最新版本的数据,当前读会对这些数据加锁,以避免其他的事务并发去修改这些数据,当前读包括:
1、加共享锁的select:select * from user lock in share mode;
2、加排它锁的select: select * from user for update;
3、增删改操作:insert、delete、update。虽然看起来增删改操作与读无关,但其实在增删改之前都要先执行当前读,以将符合条件的数据锁起来,避免其他事务并发去修改它们。
当前读为什么会有幻读问题呢?
虽然当前读能够将符合条件的数据加锁,以阻塞其他事务对于这些数据的修改,却不能够避免其他事务添加新的数据,而其他事务添加的新的数据,有可能对于当前事务的当前读造成影响:一旦其他事务所添加的数据是满足我们当前读的条件的,那么当我们在当前事务中再以同样的条件执行一次当前读时,就会读到其他事务添加的那条新的数据,幻读就发生了。分析一下原因,幻读问题的发生是因为,当前读所加的行锁只是锁住了数据库中已存在的满足条件的记录,却不能够避免其他事物添加新数据的操作。为了解决这个问题,InnoDb存储引擎引入了间隙锁。
间隙锁
同行锁锁住满足条件的行不同,间隙锁锁住的是行之间的间隙,当为某些间隙添加了间隙锁,其他事务就不能往这些间隙中添加新的数据了,比如:当前读的条件是 5<id<10,那么行锁锁住的是id落在5到10之间的行,而间隙锁锁住的是5到10之间的间隙,如果有其他事务想要添加新的数据,而新数据的id在5到10之间,那么这个添加请求将被阻塞。
可重复读在当前读模式下,用行锁+间隙锁来解决了幻读问题,行锁+间隙锁又叫Next-Key Lock。