脏写:如果一个事务修改了另一个未提交事务修改过的数据,那就意味着发生了脏写。
脏读:如果一个事务读到了另一个未提交事务修改过的数据,那就意味着发生了脏读
不可重复读:
事务两次读同一个数据结果却不一样。 和脏读的区别在于,脏读读到的是未提交的数据,不可重复读读到的是其他事务已经提交的数据。
幻读:两次查同一范围,查询到的数据数目不同(因为有另一事务操作)。
给这些问题按照严重性来排一下序:
脏写 > 脏读 > 不可重复读 > 幻读
4个隔离级别
:
-
READ UNCOMMITTED
:未提交读。 -
READ COMMITTED
:已提交读。 -
REPEATABLE READ
:可重复读。 -
SERIALIZABLE
:可串行化。
针对不同的隔离级别,并发事务可以发生不同严重程度的问题:
-
READ UNCOMMITTED
隔离级别下,可能发生脏读
、不可重复读
和幻读
问题。 -
READ COMMITTED
隔离级别下,可能发生不可重复读
和幻读
问题,但是不可以发生脏读
问题。 -
REPEATABLE READ
隔离级别下,可能发生幻读
问题,但是不可以发生脏读
和不可重复读
的问题。 -
SERIALIZABLE
隔离级别下,各种问题都不可以发生。
MySQL
的默认隔离级别为REPEATABLE READ
,我们可以手动修改一下事务的隔离级别。
MVCC原理
对于使用InnoDB
存储引擎的表来说,它的聚簇索引记录中都包含两个必要的隐藏列
-
trx_id
:每次一个事务对某条聚簇索引记录进行改动时,都会把该事务的事务id
赋值给trx_id
隐藏列。(事务ID并不是开启一个事务就会生成,仅当执行一条更新操作(增删改)才会生成!) -
roll_pointer
:每次对某条聚簇索引记录进行改动时,都会把旧的版本写入到undo日志
中,然后这个隐藏列就相当于一个指针,可以通过它来找到该记录修改前的信息。
每次对记录进行改动,都会记录一条undo日志
,每条undo日志
也都有一个roll_pointer
属性(INSERT
操作对应的undo日志
没有该属性,因为该记录并没有更早的版本),可以将这些undo日志
都连起来,串成一个链表,所以现在的情况就像下图一样:
对该记录每次更新后,都会将旧值放到一条undo日志
中,就算是该记录的一个旧版本,随着更新次数的增多,所有的版本都会被roll_pointer
属性连接成一个链表,我们把这个链表称之为版本链
,版本链的头节点就是当前记录最新的值。另外,每个版本中还包含生成该版本时对应的事务id。
ReadView
对于使用READ UNCOMMITTED
隔离级别的事务来说,由于可以读到未提交事务修改过的记录,所以直接读取记录的最新版本就好了;
对于使用SERIALIZABLE
隔离级别的事务来说,设计InnoDB
的大叔规定使用加锁的方式来访问记;
对于使用READ COMMITTED
和REPEATABLE READ
隔离级别的事务来说,都必须保证读到已经提交了的事务修改过的记录,也就是说假如另一个事务已经修改了记录但是尚未提交,是不能直接读取最新版本的记录的,核心问题就是:需要判断一下版本链中的哪个版本是当前事务可见的。为此,设计InnoDB
的大叔提出了一个ReadView
的概念,这个ReadView
中主要包含4个比较重要的内容:
-
m_ids
:表示在生成ReadView
时当前系统中活跃的读写事务的事务id
列表。 -
min_trx_id
:表示在生成ReadView
时当前系统中活跃的读写事务中最小的事务id
,也就是m_ids
中的最小值。 -
max_trx_id
:表示生成ReadView
时系统中应该分配给下一个事务的id
值。 -
creator_trx_id
:表示生成该ReadView
的事务的事务id
。
有了这个ReadView
,这样在访问某条记录时,只需要按照下边的步骤判断记录的某个版本是否可见:
-
如果被访问版本的
trx_id
属性值与ReadView
中的creator_trx_id
值相同,意味着当前事务在访问它自己修改过的记录,所以该版本可以被当前事务访问。 -
如果被访问版本的
trx_id
属性值小于ReadView
中的min_trx_id
值,表明生成该版本的事务在当前事务生成ReadView
前已经提交,所以该版本可以被当前事务访问。 -
如果被访问版本的
trx_id
属性值大于或等于ReadView
中的max_trx_id
值,表明生成该版本的事务在当前事务生成ReadView
后才开启,所以该版本不可以被当前事务访问。 -
如果被访问版本的
trx_id
属性值在ReadView
的min_trx_id
和max_trx_id
之间,那就需要判断一下trx_id
属性值是不是在m_ids
列表中,如果在,说明创建ReadView
时生成该版本的事务还是活跃的,该版本不可以被访问;如果不在,说明创建ReadView
时生成该版本的事务已经被提交,该版本可以被访问。
如果某个版本的数据对当前事务不可见的话,那就顺着版本链找到下一个版本的数据,继续按照上边的步骤判断可见性,依此类推,直到版本链中的最后一个版本。如果最后一个版本也不可见的话,那么就意味着该条记录对该事务完全不可见,查询结果就不包含该记录。
READ COMMITTED —— 每次读取数据前都生成一个ReadView
REPEATABLE READ —— 在第一次读取数据时生成一个ReadView
EAD COMMITTD
、REPEATABLE READ
这两个隔离级别的一个很大不同就是:生成ReadView的时机不同,READ COMMITTD在每一次进行普通SELECT操作前都会生成一个ReadView,而REPEATABLE READ只在第一次进行普通SELECT操作前生成一个ReadView,之后的查询操作都重复使用这个ReadView就好了。
关于purge
-
我们说
insert undo
在事务提交之后就可以被释放掉了,而update undo
由于还需要支持MVCC
,不能立即删除掉。 -
为了支持
MVCC
,对于delete mark
操作来说,仅仅是在记录上打一个删除标记,并没有真正将它删除掉。
随着系统的运行,在确定系统中包含最早产生的那个ReadView
的事务不会再访问某些update undo日志
以及被打了删除标记的记录后,有一个后台运行的purge线程
会把它们真正的删除掉。
MVCC在多事务同时存在时,SELECT语句找到是版本链上哪个版本,然后在找到的版本上返回其中所记录的数据的过程。(因为update一般都是最新版本,MVCC可以实现非锁定读select)