2020-07-16—MySQL隔离级别
前几天面试,被问到了数据库MySQL隔离级别,我自己一时间没想起来,说岔了频道,说成了事务的传播属性,现在来总结一下。
事务请看博客(https://blog.lovewinter.top/2020/07/09/spring-aop-yu-shi-wu/)
或者简书地址:https://www.jianshu.com/p/00067fa029ef 。
这次主动总结一下隔离级别方面的知识。
MySQL隔离级别的总结
事务的4个特征
事务应该具有 4 个属性:原子性、一致性、隔离性、持久性。这四个属性通常称为 ACID 特
性。
原子性(atomicity)
一致性(consistency)
隔离性(isolation)
持久性(durability)
Atomic(原子性):
事务中包括的操作被看做一个逻辑单元。这个逻辑单元中的操作要
么所有成功。要么所有失败。
Consistency(一致性):
仅仅有合法的数据能够被写入数据库,否则事务应该将其回滚到最初
状态。
Durability(持久性):
事务结束后。事务处理的结果必须可以得到固化。
Isolation(隔离性):
事务同意多个用户对同一个数据进行并发訪问,而不破坏数据的正
确性和完整性。同一时候。并行事务的改动必须与其它并行事务的改动
相互独立。
那么隔离性中就牵涉到几种隔离级别:
一个事务的执行不能被其他事务干扰。即一个事务内部的操作及使用的数据对并发的其他事
务是隔离的,并发执行的各个事务之间不能互相干扰。
(对数据库的并行执行,应该像串行执行一样)
未提交读(READ UNCOMMITED)脏读
已提交读 (READ COMMITED)不可重复读
可重复读(REPEATABLE READ)
可串行化(SERIALIZABLE)
mysql 默认的事务隔离级别为 repeatable-read
show variables like ‘%tx_isolation%’;
数据库事务的隔离级别有4个。由低到高依次为Read uncommitted、Read committed、Repeatable read、Serializable。这四个级别能够逐个解决脏读、不可反复读、幻读这几类问题。MySql设置的隔离级别默觉得Repeatable Read。可反复读级别。
隔离级别能够配置。
√: 可能出现 ×: 不会出现
脏读 | 不可反复读 | 幻读 | |
---|---|---|---|
Read uncommitted | √ | √ | √ |
Read committed | × | √ | √ |
Repeatable read | × | × | √ |
Serializable | × | × | × |
注意:我们讨论隔离级别的场景,主要是在多个事务并发的情况下。因此,接下来的解说都环绕事务并发。
Read uncommitted 读未提交
READ UNCOMMITTED是限制性最弱的隔离级别。由于该级别忽略其它事务放置的锁。使用READ UNCOMMITTED级别运行的事务,能够读取尚未由其它事务提交的改动后的数据值,这些行为称为“脏”读。我们所说的脏读,两个并发的事务,“事务A:领导给singo发工资”、“事务B:singo查询工资账户”,事务B读取了事务A尚未提交的数据。比方,事务1改动一行,事务2在事务1提交之前读取了这一行。
假设事务1回滚,事务2就读取了一行没有提交的数据。这种数据我们觉得是不存在的。
Read committed 读提交
该级别通过指定语句不能读取其它事务已改动可是尚未提交的数据值。禁止运行脏读。在当前事务中的各个语句运行之间,其它事务仍能够改动、插入或删除数据(重点是事务B仍然具有改动插入和删除的权限,所以产生不可反复读)。从而产生无法反复的读操作。或“影子”数据。比方,事务1读取了一行。事务2改动或者删除这一行而且提交。假设事务1想再一次读取这一行,它将获得改动后的数据或者发现这一样已经被删除。因此事务的第二次读取结果与第一次读取结果不同,因此也叫不可反复读。
大多数数据库的默认级别就是Read committed。比方Sql Server , Oracle。怎样解决不可反复读这一问题。请看下一个隔离级别。
不可重复读的重点是修改:
Repeatable read 反复读
REPEATABLE READ是比READ COMMITTED限制性更强的隔离级别。
该级别包含READ COMMITTED,而且另外指定了在当前事务提交之前。其它不论什么事务均不能够改动或删除当前事务已读取的数据(可以插入)。并发性低于 READ COMMITTED。由于已读数据的共享锁在整个事务期间持有,而不是在每一个语句结束时释放。
这个隔离级别仅仅是说,不可以改动和删除,可是并没有强制不能插入新的满足条件查询的数据行。
此可以得出结论:REPEATABLE READ隔离级别保证了在同样的查询条件下,同一个事务中的两个查询。第二次读取的内容肯定包换第一次读到的内容。注:Mysql的默认隔离级别就是Repeatable read。
幻读的重点在于新增或者删除。
反复读与幻读
反复读是为了保证在一个事务中,相同查询条件下读取的数据值不发生改变,可是不能保证下次相同条件查询。结果记录数不会添加。
幻读就是为了解决问题而存在的,他将这个查询范围都加锁了。所以就不能再往这个范围内插入数据。这就是SERIALIZABLE 隔离级别做的事情。
Serializable 序列化
SERIALIZABLE 是限制性最强的隔离级别,由于该级别锁定整个范围的键。并一直持有锁,直到事务完毕。该级别包含REPEATABLE READ。并添加了在事务完毕之前,其它事务不能向事务已读取的范围插入新行的限制。比方,事务1读取了一系列满足搜索条件的行。事务2在运行SQL statement产生一行或者多行满足事务1搜索条件的行时会冲突。则事务2回滚。这时事务1再次读取了一系列满足同样搜索条件的行。第二次读取的结果和第一次读取的结果同样。
**Serializable
**这个级别非常easy。读加共享锁。写加排他锁,读写相互排斥。使用的悲观锁的理论,实现简单,数据更加安全。可是并发能力非常差。假设你的业务并发的特别少或者没有并发,同一时候又要求数据及时可靠的话,能够使用这样的模式。
这里要吐槽一句,不要看到select就说不会加锁了。在Serializable这个级别,还是会加锁的。
下面是测试案例:
--事务并发问题 |
2020-07-16—MySQL隔离级别