粗谈MySQL事务的特性和隔离级别

网上对于此类的文章已经十分饱和了,那还写的原因很简单——作为自己的理解笔记。

前言

​  此篇文章作为自己学习MySQL的一些个人理解,使用的引擎是InnoDb。首先先讲讲事务的概念,在《高性能MySQL》第三版中其对事务的描述是这样的:

事务就是一组原子性的SQL查询,或者说一个独立的工作单元。如果数据库引擎能够成功地对数据库应用该组查询的全部语句,那么就执行该组查询。如果其中有任何一条语句因为崩溃或其他原因无法执行,那么所有的语句都不会执行。

​​  换句话说,事务就是一个整体单位,里面的SQL语句不会单独执行,就像某些商品一般,由多个组件组成,但是我绝对不单独卖组件,要买就买整个商品,不然就不卖。

​​  简单的理解了事务之后,还需要知道事务的目的就是为了保证数据的正确性和一致性,那么为此则诞生出其4个特性(后面再细讲),而为了实现这四个特性又需要许多具体的实现,其中就包括为了隔离性而产生的四个隔离级别,这四种隔离级别又产生了三个问题(脏读、不可重复读和幻读),这就是其大致的关系,接下来让我们来看看这些具体到底是个什么东西。

1 四种特性(ACID)

​​  说起事务的特性,那肯定张口就来ACID,然而除了ACID之外我们还是需要说点其他东西的。

​​  原子性(Atomicity):意思是说一个事务应当作为一个不可分割的最小单位,整个事务的操作要么全部执行成功要么全部不执行,像原子一样不可分割(别跟我谈夸克),这里的执行是指执行成功,如果有一个操作执行失败了那么就全部不执行,这也是我们平时见到的回滚。

​​  一致性(Consistency):书上给出的意思是事务总是从一个一致性的状态跳到另一个一致性的状态。我的理解是在涉及到的数据范围内是守恒的,也就是说,整体的数据是不变的,拿万能的转钱例子来说,A账户转给B``200元,那么由A和B组成的这个数据范围来说数据并没有发生改变(-200+200=0),只是数据的组成方式变化了,所以是从一个一致性状态—>另一个一致性状态。

​​  隔离性(Isolation):通常来说,一个事务的操作对于其他的事务的不可见的,也就是说一般而言事务都是独立的。但是这跟数据库的隔离级别有关,除了某个(没错,就是你——读未提交同学)隔离级别之外,其他的都是不可见的,而这种事务可见的级别很少用到,所以说的是'通常来说'。

​​  持久性(Durability):事务一旦完成,那么该事务引起的数据变化将永久生效,不会改变(除非被另外一个事务改动)。不过书上提到这其实跟实行的策略相关,但这貌似就有点走远了(是的,我不懂!)。

​​  以上就是事务的四种特性,然而其中隔离性的实现则是要看数据库的隔离级别。

2 数据库的隔离级别

​​  在MySQL中隔离级别有四种,每种隔离级别对应的事务体现不同,可能出现的问题也各自不同。

​​  未提交读(read uncommited):在这个隔离级别中,在一个事务执行的操作就算不提交也能被其他的事务看到。在这个级别中一个事务可能读到其他事务还没提交的脏数据,即可能出现脏读。如下图所示,序号表示执行的顺序。

img

​​  可以看到,在界面1的事务中往test表插入了一条数据,此时就算还没提交在页面2的另一个事务中也可以看到提交的数据。

​​  提交读(read commited):在一个事务提交之后,其他事务才可以看到事务的修改。此隔离级别可能会出现同一个事务中执行相同的查询却读到不同的数据,即不可重复读(nonrepeatable read),另未提交读也可能出现不可重复读。例子如下

img

​​  可重复读(repeatable read):这是MySQL的默认隔离级别在事务开始的时候会保存此刻的一个快照(这里啰嗦一下,实际上是开启事务后执行第一条语句的时候准备的快照,准备快照的方法则是记录当前事务的版本号,没有进行数据的复制,不明白事务版本号或隐藏字段的可以看看MySQL的MVCC),然后接下来这个事务的所有数据读取都是从这个快照读,所以不会出现不可重复读的情况,但是还是有可能出现幻读。意思就是读取的是快照表数据不会变化,但是进行写操作如更新的时候更新的数量可能会跟预期的不同。如图

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/zwfjpy.html