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

58次阅读
没有评论

共计 6025 个字符,预计需要花费 16 分钟才能阅读完成。

自动写代码机器人,免费开通

丸趣 TV 小编给大家分享一下 MySQL 事务隔离级别以及脏读、幻读、不可重复读的示例,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!

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

推荐(免费):mysql 视频教程

事务的隔离性

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 存储引擎中,可以使用以下命令来设置全局或者当前会话的事务隔离级别:

SET [GLOBAL|SESSION] TRANSACTION ISOLATION LEVEL{ READ UNCOMMITTED
 | READ COMMITTED
 | REPEATABLE READ
 | SERIALIZABLE}

如想设置当前会话的隔离级别为读提交,可以使用如下语句:

SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;

如果想在 MySQL 数据库启动时就设置事务的默认隔离级别,那就需要修改配置文件中 transaction-isolation 的值,比方说,我们在启动前指定了 transaction-isolation = READ COMMITTED,那么事务的默认隔离级别就从原来的 REPEATABLE READ 变成了 READ COMMITTED。

查看当前会话的事务隔离级别,可以用如下语句:

SELECT @@transaction_isolation;

查看全局的事务隔离级别,可以使用如下语句:

SELECT @@global.transaction_isolation;

注意:transaction_isolation 是在 MySQL 5.7.20 的版本中引入来替换 tx_isolation 的,如果你使用的是之前版本的 MySQL,请将上述用到的 transaction_isolation 的地方替换为 tx_isolation。

事务的隔离性

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 存储引擎中,可以使用以下命令来设置全局或者当前会话的事务隔离级别:

SET [GLOBAL|SESSION] TRANSACTION ISOLATION LEVEL{ READ UNCOMMITTED
 | READ COMMITTED
 | REPEATABLE READ
 | SERIALIZABLE}

如想设置当前会话的隔离级别为读提交,可以使用如下语句:

SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;

如果想在 MySQL 数据库启动时就设置事务的默认隔离级别,那就需要修改配置文件中 transaction-isolation 的值,比方说,我们在启动前指定了 transaction-isolation = READ COMMITTED,那么事务的默认隔离级别就从原来的 REPEATABLE READ 变成了 READ COMMITTED。

查看当前会话的事务隔离级别,可以用如下语句:

SELECT @@transaction_isolation;

查看全局的事务隔离级别,可以使用如下语句:

SELECT @@global.transaction_isolation;

注意:transaction_isolation 是在 MySQL 5.7.20 的版本中引入来替换 tx_isolation 的,如果你使用的是之前版本的 MySQL,请将上述用到的 transaction_isolation 的地方替换为 tx_isolation。

以上是“MySQL 事务隔离级别以及脏读、幻读、不可重复读的示例”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注丸趣 TV 行业资讯频道!

向 AI 问一下细节

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2023-12-16发表,共计6025字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)