请选择 进入手机版 | 继续访问电脑版

MySQL事务隔离级别以及脏读、幻读、不可重复读示例

[复制链接]
滚雪球少年 发表于 2021-1-2 19:43:16 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题
事务的隔离性

MySQL是一个客户端/服务器架构的软件,对于同一个服务器来说,可以有若干个客户端与之毗连,每个客户端与服务器毗连上之后,就可以称之为一个会话(Session)。每个客户端都可以在自己的会话中向服务器发出请求语句,一个请求语句大概是某个事务的一部门,也就是对于服务器来说大概同时处置惩罚多个事务。当数据库上有多个事务同时执行的时候,就大概出现脏读(Dirty Read)、不可重复读(Non-Repeatable Read)、幻读(Phantom Read)的问题,为了办理这些问题,就有了 “隔离级别” 的概念。
理论上在某个事务对某个数据举行访问时,其他事务应该举行列队,当该事务提交之后,其他事务才可以继续访问这个数据。但一般情况下隔离得越严实,效率就会越低。因此许多时候,我们都要在隔离性和效率二者之间寻找一个均衡点。
事务并发执行遇到的问题

脏读(Dirty Read): 脏读是指一个事务读到了另一个未提交事务修改过的数据。
如小王的账户中有100的余额,接下来有两个事务对小王的账户举行访问。
会话A会话Bbegin;update xxx set balance = balance+50 where client_no = ‘小王客户号’ ;begin;select balance from xxx where client_no = ‘小王客户号’ ;
(如果读到150,则意味着发生了脏读)rollback;commit;如上,会话A和会话B各开启了一个事务,会话A先给小王账户余额加了50,此时账户B查询小王账户余额为150,接下来会话A举行了回滚,那会话B查询到的150就成一个不正确的脏数据。
不可重复读(Non-Repeatable Read): 不可重复读是指在同一个事务内多次读取同一数据聚集,但查到的效果却不相同。发生不可重复读的原因是在多次搜索期间查询的数据被别的事务修改了。
看如下的两个会话请求。
会话A会话Bbegin;select balance from xxx where client_no = ‘小王客户号’ ;
(读到余额为100)begin;update xxx set balance = balance+50 where client_no = ‘小王客户号’ ;commit;select balance from xxx where client_no = ‘小王客户号’ ;
(如果读到150,则意味着发生了不可重复读)commit;在会话A的同一个事务中,两次相同查询的效果差别,意味着发生了不可重复读。
幻读(Phantom Read): 所谓幻读,指的是当某个事务在读取某个范围内的记载时,别的一个事务又在该范围内插入了新的记载,当之前的事务再次读取该范围的记载时,会读取到之前没有读到的数据。
如果账户表中现在只有小王的余额为100,再看下如下的两个会话请求。
会话A会话Bbegin;select name from xxx where balance = 100 ;
(读到name为‘小王’)begin;insert into xxx(client_no,name,balance) values(‘小张客户号’,‘小张’,100);commit;select name from xxx where balance = 100 ;
(如果读到了‘小王’和‘小张’,则意味着发生了幻读)commit;会话A事务中的第二次查询,查到了第一次查询没有查到的 name ‘小张’,这就意味着出现了幻读。
SQL尺度制定的四种隔离级别

ISO 和 ANIS SQL 尺度制定了四种事务隔离级别的尺度,分别为:读未提交(read uncommitted)、读提交(read committed)、可重复读(repeatable read)和串行化(serializable )。
我们先来看下这四种隔离级别的意思。


  • 读未提交: 一个事务还没提交时,它做的变更就能被别的事务看到。
  • 读提交: 一个事务提交之后,它做的变更才会被其他事务看到。
  • 可重复读: 一个事务执行过程中看到的数据,总是跟这个事务在启动时看到的数据是一致的。固然在可重复读隔离级别下,未提交的变更对其他事务也是不可见的。
  • 串行化: 顾名思义是对于同一行记载,“写”会加“写锁”,“读”会加“读锁”。当出现读写锁辩说的时候,后访问的事务必须等前一个事务执行完成,才华继续执行。
SQL 尺度中规定,针对差别的隔离级别,并发事务可以发生差别严重水平的问题,详细情况如下:
( √ 表现可以发生;× 表现不可以发生)
隔离级别脏读不可重复读幻读读未提交(read uncommitted)√√√读提交(read committed)×√√可重复读(repeatable read)××√串行化(serializable )×××MySQL对四种隔离级别的支持情况

虽然 ISO 和 ANIS SQL 尺度制定了四种事务隔离级别的尺度,但不是所有数据库厂商都遵循这些尺度,比如 Oracle 数据库就不支持读未提交(read uncommitted)和可重复读(repeatable read)的事务隔离级别。
MySQL InnoDB 存储引擎支持4种隔离级别,但与 SQL 尺度中界说的差别的是,InnoDB 存储引擎在默认的可重复读(repeatable read)事务隔离级别下,使用 Next-Key Lock 锁的算法,制止了幻读的产生。也就是说 InnoDB 存储引擎在可重复读(repeatable read)的事务隔离级别下,已经可以完全包管事务的隔离性要求,即到达了 SQL 尺度中的串行化(serializable )隔离级别的要求。
如何设置事务的隔离级别

在 InnoDB 存储引擎中,可以使用以下下令来设置全局大概当前会话的事务隔离级别:
  1. SET [GLOBAL|SESSION] TRANSACTION ISOLATION LEVEL{        READ UNCOMMITTED        | READ COMMITTED        | REPEATABLE READ        | SERIALIZABLE}
复制代码
如想设置当前会话的隔离级别为读提交,可以使用如下语句:
  1. SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
复制代码
如果想在 MySQL 数据库启动时就设置事务的默认隔离级别,那就需要修改设置文件中 transaction-isolation 的值,比方说,我们在启动前指定了 transaction-isolation = READ COMMITTED,那么事务的默认隔离级别就从原来的 REPEATABLE READ 变成了READ COMMITTED。
检察当前会话的事务隔离级别,可以用如下语句:
  1. SELECT @@transaction_isolation;
复制代码
检察全局的事务隔离级别,可以使用如下语句:
  1. SELECT @@global.transaction_isolation;
复制代码
注意:transaction_isolation 是在 MySQL 5.7.20 的版本中引入来替换tx_isolation的,如果你使用的是之前版本的 MySQL,请将上述用到的 transaction_isolation 的地方替换为 tx_isolation 。

来源:https://blog.csdn.net/daidaineteasy/article/details/112055945
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

发布主题

专注素材教程免费分享
全国免费热线电话

18768367769

周一至周日9:00-23:00

反馈建议

27428564@qq.com 在线QQ咨询

扫描二维码关注我们

Powered by Discuz! X3.4© 2001-2013 Comsenz Inc.( 蜀ICP备2021001884号-1 )