写点什么

MySQL- 技术专题 - 知识点介绍

发布于: 2021 年 03 月 27 日
MySQL-技术专题-知识点介绍

1.总体知识图谱

2.什么是事务

事务的定义是:事务是数据库管理系统(DBMS)执行过程中的一个逻辑单位,由一个有限的数据库操作序列构成。

事务四大特性


事务包含四大特性,即原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)和持久性(Durability)(ACID)。


  • 原子性(Atomicity) 原子性是指对数据库的一系列操作,要么全部成功,要么全部失败,不可能出现部分成功的情况。以转账场景为例,一个账户的余额减少,另一个账户的余额增加,这两个操作一定是同时成功或者同时失败的。


  • 一致性(Consistency) 一致性是指数据库的完整性约束没有被破坏,在事务执行前后都是合法的数据状态。这里的一致可以表示数据库自身的约束没有被破坏,比如某些字段的唯一性约束、字段长度约束等等;还可以表示各种实际场景下的业务约束,比如上面转账操作,一个账户减少的金额和另一个账户增加的金额一定是一样的。


  • 隔离性(Isolation) 隔离性指的是多个事务彼此之间是完全隔离、互不干扰的。隔离性的最终目的也是为了保证一致性。


  • 持久性(Durability) 持久性是指只要事务提交成功,那么对数据库做的修改就被永久保存下来了,不可能因为任何原因再回到原来的状态。


事务基本状态

根据事务所处的不同阶段,事务大致可以分为以下 5 个状态:

  • 活动的(active) 当事务对应的数据库操作正在执行过程中,则该事务处于活动状态。

  • 部分提交的(partially committed) 当事务中的最后一个操作执行完成,但还未将变更刷新到磁盘时,则该事务处于部分提交状态。

  • 失败的(failed) 当事务处于活动或者部分提交状态时,由于某些错误导致事务无法继续执行,则事务处于失败状态。

  • 中止的(aborted) 当事务处于失败状态,且回滚操作执行完毕,数据恢复到事务执行之前的状态时,则该事务处于中止状态。

  • 提交的(committed) 当事务处于部分提交状态,并且将修改过的数据都同步到磁盘之后,此时该事务处于提交状态。

事务隔离级别

前面提到过,事务必须具有隔离性。实现隔离性最简单的方式就是不允许事务并发,每个事务都排队执行,但是这种方式性能实在太差了。为了兼顾事务的隔离性和性能,事务支持不同的隔离级别。

为了方便表述后续的内容,我们先建一张示例表 hero。

事务并发执行

  • 脏写(Dirty Write) 脏写是指一个事务修改了其它事务未提交的数据。

Session A 和 Session B 各开启了一个事务,Session B 中的事务先将 number 列为 1 的记录的 name 列更新为'关羽',然后 Session A 中的事务接着又把这条 number 列为 1 的记录的 name 列更新为张飞。如果之后 Session B 中的事务进行了回滚,那么 Session A 中的更新也将不复存在,这种现象就称之为脏写

  • 脏读(Dirty Read) 脏读是指一个事务读到了其它事务未提交的数据。

如上图,Session A 和 Session B 各开启了一个事务,Session B 中的事务先将 number 列为 1 的记录的 name 列更新为'关羽',然后 Session A 中的事务再去查询这条 number 为 1 的记录,如果读到列 name 的值为'关羽',而 Session B 中的事务稍后进行了回滚,那么 Session A 中的事务相当于读到了一个不存在的数据,这种现象就称之为脏读。

  • 不可重复读(Non-Repeatable Read) 不可重复读指的是在一个事务执行过程中,读取到其它事务已提交的数据,导致两次读取的结果不一致。

如上图,我们在 Session B 中提交了几个隐式事务(mysql 会自动为增删改语句加事务),这些事务都修改了 number 列为 1 的记录的列 name 的值,每次事务提交之后,如果 Session A 中的事务都可以查看到最新的值,这种现象也被称之为不可重复读。

  • 幻读(Phantom) 幻读是指的是在一个事务执行过程中,读取到了其他事务新插入数据,导致两次读取的结果不一致。

如上图,Session A 中的事务先根据条件 number > 0 这个条件查询表 hero,得到了 name 列值为'刘备'的记录;之后 Session B 中提交了一个隐式事务,该事务向表 hero 中插入了一条新记录;之后 Session A 中的事务再根据相同的条件 number > 0 查询表 hero,得到的结果集中包含 Session B 中的事务新插入的那条记录,这种现象也被称之为幻读。


不可重复读和幻读的区别在于不可重复读是读到的是其他事务修改或者删除的数据,而幻读读到的是其它事务新插入的数据。


脏写的问题太严重了,任何隔离级别都必须避免。其它无论是脏读,不可重复读,还是幻读,它们都属于数据库的读一致性的问题,都是在一个事务里面前后两次读取出现了不一致的情况。

四种隔离级别

在 SQL 标准中设立了 4 种隔离级别,用来解决上面的读一致性问题。不同的隔离级别可以解决不同的读一致性问题。

  • READ UNCOMMITTED:未提交读。

  • READ COMMITTED:已提交读。

  • REPEATABLE READ:可重复读。

  • SERIALIZABLE:串行化。

各个隔离级别下可能出现的读一致性问题如下:


InnoDB 支持四个隔离级别(和 SQL 标准定义的基本一致)。隔离级别越高,事务的并发度就越低。唯一的区别就在于,InnoDB 在可重复读(REPEATABLE READ)的级别就解决了幻读的问题。这也是 InnoDB 使用可重复读 作为事务默认隔离级别的原因。

 MVCC

MVCC(Multi Version Concurrency Control),中文名是多版本并发控制,简单来说就是通过维护数据历史版本,从而解决并发访问情况下的读一致性问题。

版本链

在 InnoDB 中,每行记录实际上都包含了两个隐藏字段:事务 id(trx_id)和回滚指针(roll_pointer)。

  1. trx_id:事务 id。每次修改某行记录时,都会把该事务的事务 id 赋值给 trx_id 隐藏列。

  2. roll_pointer:回滚指针。每次修改某行记录时,都会把 undo 日志地址赋值给 roll_pointer 隐藏列。

假设 hero 表中只有一行记录,当时插入的事务 id 为 80。此时,该条记录的示例图如下:

假设之后两个事务 id 分别为 100、200 的事务对这条记录进行 UPDATE 操作,操作流程如下:

由于每次变动都会先把 undo 日志记录下来,并用 roll_pointer 指向 undo 日志地址。因此可以认为,对该条记录的修改日志串联起来就形成了一个版本链,版本链的头节点就是当前记录最新的值。如下:

ReadView

如果数据库隔离级别是未提交读(READ UNCOMMITTED),读取版本链中最新版本的记录即可。

如果是是串行化(SERIALIZABLE),事务之间是加锁执行的,不存在读不一致的问题。

如果是已提交读(READ COMMITTED)或者可重复读(REPEATABLE READ),就需要遍历版本链中的每一条记录,判断该条记录是否对当前事务可见,直到找到为止(遍历完还没找到就说明记录不存在)。

InnoDB 通过 ReadView 实现了这个功能。ReadView 中主要包含以下 4 个内容:

  • m_ids:表示在生成 ReadView 时当前系统中活跃的读写事务的事务 id 列表。

  • mintrxid:表示在生成 ReadView 时当前系统中活跃的读写事务中最小的事务 id,也就是 m_ids 中的最小值。

  • maxtrxid:表示生成 ReadView 时系统中应该分配给下一个事务的 id 值。

  • creatortrxid:表示生成该 ReadView 事务的事务 id。


如果被访问版本的 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 时生成该版本的事务已经被提交,该版本可以被访问。


在 MySQL 中,READ COMMITTED 和 REPEATABLE READ 隔离级别的的一个非常大的区别就是它们生成 ReadView 的时机不同。READ COMMITTED 在每次读取数据前都会生成一个 ReadView,这样就能保证每次都能读到其它事务已提交的数据。REPEATABLE READ 只在第一次读取数据时生成一个 ReadView,这样就能保证后续读取的结果完全一致。

事务并发访问同一数据资源的情况主要就分为读-读、写-写和读-写三种。


  • 读-读 即并发事务同时访问同一行数据记录。由于两个事务都进行只读操作,不会对记录造成任何影响,因此并发读完全允许。


  • 写-写 即并发事务同时修改同一行数据记录。这种情况下可能导致脏写问题,这是任何情况下都不允许发生的,因此只能通过加锁实现,也就是当一个事务需要对某行记录进行修改时,首先会先给这条记录加锁,如果加锁成功则继续执行,否则就排队等待,事务执行完成或回滚会自动释放锁。


  • 读-写 即一个事务进行读取操作,另一个进行写入操作。这种情况下可能会产生脏读、不可重复读、幻读。最好的方案是读操作利用多版本并发控制(MVCC),写操作进行加锁。


锁的粒度

按锁作用的数据范围进行分类的话,锁可以分为行级锁和表级锁。

  1. 行级锁:作用在数据行上,锁的粒度比较小。

  2. 表级锁:作用在整张数据表上,锁的粒度比较大。

锁的分类

为了实现读-读之间不受影响,并且写-写、读-写之间能够相互阻塞,Mysql 使用了读写锁的思路进行实现,具体来说就是分为了共享锁和排它锁:

1.共享锁(Shared Locks):简称 S 锁,在事务要读取一条记录时,需要先获取该记录的 S 锁。S 锁可以在同一时刻被多个事务同时持有。我们可以用 select ...... lock in share mode;的方式手工加上一把 S 锁。

2.排他锁(Exclusive Locks):简称 X 锁,在事务要改动一条记录时,需要先获取该记录的 X 锁。X 锁在同一时刻最多只能被一个事务持有。X 锁的加锁方式有两种,第一种是自动加锁,在对数据进行增删改的时候,都会默认加上一个 X 锁。还有一种是手工加锁,我们用一个 FOR UPDATE 给一行数据加上一个 X 锁。

还需要注意的一点是,如果一个事务已经持有了某行记录的 S 锁,另一个事务是无法为这行记录加上 X 锁的,反之亦然。

除了共享锁(Shared Locks)和排他锁(Exclusive Locks),Mysql 还有意向锁(Intention Locks)。意向锁是由数据库自己维护的,一般来说,当我们给一行数据加上共享锁之前,数据库会自动在这张表上面加一个意向共享锁(IS 锁);当我们给一行数据加上排他锁之前,数据库会自动在这张表上面加一个意向排他锁(IX 锁)。意向锁可以认为是 S 锁和 X 锁在数据表上的标识,通过意向锁可以快速判断表中是否有记录被上锁,从而避免通过遍历的方式来查看表中有没有记录被上锁,提升加锁效率。例如,我们要加表级别的 X 锁,这时候数据表里面如果存在行级别的 X 锁或者 S 锁的,加锁就会失败,此时直接根据意向锁就能知道这张表是否有行级别的 X 锁或者 S 锁。

InnoDB 中的表级锁

InnoDB 中的表级锁主要包括表级别的意向共享锁(IS 锁)和意向排他锁(IX 锁)以及自增锁(AUTO-INC 锁)。其中 IS 锁和 IX 锁在前面已经介绍过了,这里不再赘述,我们接下来重点了解一下 AUTO-INC 锁。

大家都知道,如果我们给某列字段加了 AUTO_INCREMENT 自增属性,插入的时候不需要为该字段指定值,系统会自动保证递增。系统实现这种自动给 AUTO_INCREMENT 修饰的列递增赋值的原理主要是两个:

1.AUTO-INC 锁:在执行插入语句的时先加上表级别的 AUTO-INC 锁,插入执行完成后立即释放锁。如果我们的插入语句在执行前无法确定具体要插入多少条记录,比如 INSERT ... SELECT 这种插入语句,一般采用 AUTO-INC 锁的方式。

2.轻量级锁:在插入语句生成 AUTO_INCREMENT 值时先才获取这个轻量级锁,然后在 AUTO_INCREMENT 值生成之后就释放轻量级锁。如果我们的插入语句在执行前就可以确定具体要插入多少条记录,那么一般采用轻量级锁的方式对 AUTO_INCREMENT 修饰的列进行赋值。这种方式可以避免锁定表,可以提升插入性能。

“mysql 默认根据实际场景自动选择加锁方式,当然也可以通过 innodb_autoinc_lock_mode 强制指定只使用其中一种。”

InnoDB 中的行级锁

前面说过,通过 MVCC 可以解决脏读、不可重复读、幻读这些读一致性问题,但实际上这只是解决了普通 select 语句的数据读取问题。事务利用 MVCC 进行的读取操作称之为快照读,所有普通的 SELECT 语句在 READ COMMITTED、REPEATABLE READ 隔离级别下都算是快照读。除了快照读之外,还有一种是锁定读,即在读取的时候给记录加锁,在锁定读的情况下依然要解决脏读、不可重复读、幻读的问题。由于都是在记录上加锁,这些锁都属于行级锁。

InnoDB 的行锁,是通过锁住索引来实现的,如果加锁查询的时候没有使用过索引,会将整个聚簇索引都锁住,相当于锁表了。根据锁定范围的不同,行锁可以使用记录锁(Record Locks)、间隙锁(Gap Locks)和临键锁(Next-Key Locks)的方式实现。假设现在有一张表 t,主键是 id。我们插入了 4 行数据,主键值分别是 1、4、7、10。接下来我们就以聚簇索引为例,具体介绍三种形式的行锁。

  • 记录锁(Record Locks) 所谓记录,就是指聚簇索引中真实存放的数据,比如上面的 1、4、7、10 都是记录。


显然,记录锁就是直接锁定某行记录。当我们使用唯一性的索引(包括唯一索引和聚簇索引)进行等值查询且精准匹配到一条记录时,此时就会直接将这条记录锁定。例如 select * from t where id =4 for update;就会将 id=4 的记录锁定。

  • 间隙锁(Gap Locks) 间隙指的是两个记录之间逻辑上尚未填入数据的部分,比如上述的(1,4)、(4,7)等。


同理,间隙锁就是锁定某些间隙区间的。当我们使用用等值查询或者范围查询,并且没有命中任何一个 record,此时就会将对应的间隙区间锁定。例如 select * from t where id =3 for update;或者 select * from t where id > 1 and id < 4 for update;就会将(1,4)区间锁定。

  • 临键锁(Next-Key Locks) 临键指的是间隙加上它右边的记录组成的左开右闭区间。比如上述的(1,4]、(4,7]等。


临键锁就是记录锁(Record Locks)和间隙锁(Gap Locks)的结合,即除了锁住记录本身,还要再锁住索引之间的间隙。当我们使用范围查询,并且命中了部分 record 记录,此时锁住的就是临键区间。注意,临键锁锁住的区间会包含最后一个 record 的右边的临键区间。例如 select * from t where id > 5 and id <= 7 for update;会锁住(4,7]、(7,+∞)。mysql 默认行锁类型就是临键锁(Next-Key Locks)。当使用唯一性索引,等值查询匹配到一条记录的时候,临键锁(Next-Key Locks)会退化成记录锁;没有匹配到任何记录的时候,退化成间隙锁。

间隙锁(Gap Locks)和临键锁(Next-Key Locks)都是用来解决幻读问题的,在已提交读(READ COMMITTED)隔离级别下,间隙锁(Gap Locks)和临键锁(Next-Key Locks)都会失效!


用户头像

我们始于迷惘,终于更高的迷惘。 2020.03.25 加入

一个酷爱计算机技术、健身运动、悬疑推理的极客狂人,大力推荐安利Java官方文档:https://docs.oracle.com/javase/specs/index.html

评论

发布
暂无评论
MySQL-技术专题-知识点介绍