简述MySql四种事务隔离级别
隔离级别:
隔离性其实比想象的要复杂。在SQL标准中定义了四种隔离级别,每一个事务中所做的修改,哪些在事务内和事务间是可见的,哪些是不可见的。较低级别的隔离通常可以执行更高的并发,系统的开销也更低。
下面简单地介绍一下四种隔离级别。
1.READUNCOMMITTED(未提交读)
在READUNCOMMITTED级别,事务中的修改,即使没有提交,对其他事务也都是可见的。事务可以读取未提交的数据,这也被称为脏读(DirtyRead).这个级别会导致很多问题,从性能上来说,READUNCOMMITTED不会比其他的级别好太多,但却缺乏其他级别的很多好处,除非真的有非常必要的理由,在实际应用中一般很少使用。
(1)所有事务都可以看到其他未提交事务的执行结果
(2)本隔离级别很少用于实际应用,因为它的性能也不比其他级别好多少
(3)该级别引发的问题是——脏读(DirtyRead):读取到了未提交的数据
#首先,修改隔离级别 settx_isolation='READ-UNCOMMITTED'; select@@tx_isolation; +------------------+ |@@tx_isolation| +------------------+ |READ-UNCOMMITTED| +------------------+#事务A:启动一个事务 starttransaction; select*fromtx; +------+------+ |id|num| +------+------+ |1|1| |2|2| |3|3| +------+------+#事务B:也启动一个事务(那么两个事务交叉了) 在事务B中执行更新语句,且不提交 starttransaction; updatetxsetnum=10whereid=1; select*fromtx; +------+------+ |id|num| +------+------+ |1|10| |2|2| |3|3| +------+------+#事务A:那么这时候事务A能看到这个更新了的数据吗? select*fromtx; +------+------+ |id|num| +------+------+ |1|10|--->可以看到!说明我们读到了事务B还没有提交的数据 |2|2| |3|3| +------+------+#事务B:事务B回滚,仍然未提交 rollback; select*fromtx; +------+------+ |id|num| +------+------+ |1|1| |2|2| |3|3| +------+------+#事务A:在事务A里面看到的也是B没有提交的数据 select*fromtx; +------+------+ |id|num| +------+------+ |1|1|--->脏读意味着我在这个事务中(A中),事务B虽然没有提交,但它任何一条数据变化,我都可以看到! |2|2| |3|3| +------+------+
2.READCOMMITTED(提交读)
大多数数据库系统的默认隔离级别都是READCOMMITTED,MySQL不是.READCOMMITTED满足前面提到的隔离性的简单定义:一个事务开始时,只能“看见”已经提交的事务所做的修改。换句话说,一个事务从开始直到提交之前,所做的任何修改对其他事务都是不可见的。这个级别有时候也叫做不可重复读(nonrepeatableread),因为两次执行同样的查询,可能会得到不一样的结果。
(1)这是大多数数据库系统的默认隔离级别(但不是MySQL默认的)
(2)它满足了隔离的简单定义:一个事务只能看见已经提交事务所做的改变
(3)这种隔离级别出现的问题是——不可重复读(NonrepeatableRead):不可重复读意味着我们在同一个事务中执行完全相同的select语句时可能看到不一样的结果。
导致这种情况的原因可能有:
(1)有一个交叉的事务有新的commit,导致了数据的改变;
(2)一个数据库被多个实例操作时,同一事务的其他实例在该实例处理其间可能会有新的commit
#首先修改隔离级别 settx_isolation='read-committed'; select@@tx_isolation; +----------------+ |@@tx_isolation| +----------------+ |READ-COMMITTED| +----------------+#事务A:启动一个事务 starttransaction; select*fromtx; +------+------+ |id|num| +------+------+ |1|1| |2|2| |3|3| +------+------+#事务B:也启动一个事务(那么两个事务交叉了) 在这事务中更新数据,且未提交 starttransaction; updatetxsetnum=10whereid=1; select*fromtx; +------+------+ |id|num| +------+------+ |1|10| |2|2| |3|3| +------+------+#事务A:这个时候我们在事务A中能看到数据的变化吗? select*fromtx;---------------> +------+------+| |id|num|| +------+------+| |1|1|--->并不能看到!| |2|2|| |3|3|| +------+------+|——>相同的select语句,结果却不一样 | #事务B:如果提交了事务B呢?| commit;| | #事务A:| select*fromtx;---------------> +------+------+ |id|num| +------+------+ |1|10|--->因为事务B已经提交了,所以在A中我们看到了数据变化 |2|2| |3|3| +------+------+
3.REPEATABLEREAD(可重复读)
REPEATABLEREAD解决了脏读的问题。该级别保证了在同一个事务中多次读取同样记录的结果是一致的。但是理论上,可重复读隔离级别还是无法解决另外一个幻读(PhantomRead)的问题。所谓幻读,指的是当某个事务在读取某个范围内的记录时,另外一个事务又在该范围内插人了新的记录,当之前的事务再次读取该范围的记录时,会产生幻行(PhantomRow).InnoDB和XtraDB存储引擎通过多版本并发控制(MVCC,MultiversionConcurrencyControl)解决了幻读的问题。
(1)这是MySQL的默认事务隔离级别
(2)它确保同一事务的多个实例在并发读取数据时,会看到同样的数据行
(3)此级别可能出现的问题——幻读(PhantomRead):当用户读取某一范围的数据行时,另一个事务又在该范围内插入了新行,当用户再读取该范围的数据行时,会发现有新的幻影行
(4)InnoDB和Falcon存储引擎通过多版本并发控制(MVCC,MultiversionConcurrencyControl)机制解决了该问题
#首先,更改隔离级别 settx_isolation='repeatable-read'; select@@tx_isolation; +-----------------+ |@@tx_isolation| +-----------------+ |REPEATABLE-READ| +-----------------+#事务A:启动一个事务 starttransaction; select*fromtx; +------+------+ |id|num| +------+------+ |1|1| |2|2| |3|3| +------+------+#事务B:开启一个新事务(那么这两个事务交叉了) 在事务B中更新数据,并提交 starttransaction; updatetxsetnum=10whereid=1; select*fromtx; +------+------+ |id|num| +------+------+ |1|10| |2|2| |3|3| +------+------+ commit;#事务A:这时候即使事务B已经提交了,但A能不能看到数据变化? select*fromtx; +------+------+ |id|num| +------+------+ |1|1|--->还是看不到的!(这个级别2不一样,也说明级别3解决了不可重复读问题) |2|2| |3|3| +------+------+#事务A:只有当事务A也提交了,它才能够看到数据变化 commit; select*fromtx; +------+------+ |id|num| +------+------+ |1|10| |2|2| |3|3| +------+------+
4.SERIALIZABLE(可串行化)
SERIALIZABLE是最高的隔离级别。它通过强制事务串行执行,避免了前面说的幻读的问题.简单来说,SERIALIZABLE会在读取的每一行数据上都加锁,所以可能导致大量的超时和锁争用的问题。实际应用中也很少用到这个隔离级别,只有在非常需要确保数据的一致性而且可以接受没有并发的情况下,才考虑采用该级别。
(1)这是最高的隔离级别
(2)它通过强制事务排序,使之不可能相互冲突,从而解决幻读问题。简言之,它是在每个读的数据行上加上共享锁。
(3)在这个级别,可能导致大量的超时现象和锁竞争
#首先修改隔离界别 settx_isolation='serializable'; select@@tx_isolation; +----------------+ |@@tx_isolation| +----------------+ |SERIALIZABLE| +----------------+#事务A:开启一个新事务 starttransaction;#事务B:在A没有commit之前,这个交叉事务是不能更改数据的 starttransaction; inserttxvalues('4','4'); ERROR1205(HY000):Lockwaittimeoutexceeded;tryrestartingtransaction updatetxsetnum=10whereid=1; ERROR1205(HY000):Lockwaittimeoutexceeded;tryrestartingtransaction
以上就是简述MySql四种隔离级别的详细内容,更多关于MySQL隔离级别的资料请关注毛票票其它相关文章!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。