千家信息网

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

发表于:2025-02-02 作者:千家信息网编辑
千家信息网最后更新 2025年02月02日,小编给大家分享一下MySQL事务隔离级别以及脏读、幻读、不可重复读的示例,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!推荐
千家信息网最后更新 2025年02月02日MySQL事务隔离级别以及脏读、幻读、不可重复读的示例

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

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

事务的隔离性

MySQL是一个客户端/服务器架构的软件,对于同一个服务器来说,可以有若干个客户端与之连接,每个客户端与服务器连接上之后,就可以称之为一个会话(Session)。每个客户端都可以在自己的会话中向服务器发出请求语句,一个请求语句可能是某个事务的一部分,也就是对于服务器来说可能同时处理多个事务。当数据库上有多个事务同时执行的时候,就可能出现脏读(Dirty Read)、不可重复读(Non-Repeatable Read)、幻读(Phantom Read)的问题,为了解决这些问题,就有了 "隔离级别" 的概念。

理论上在某个事务对某个数据进行访问时,其他事务应该进行排队,当该事务提交之后,其他事务才可以继续访问这个数据。但一般情况下隔离得越严实,效率就会越低。因此很多时候,我们都要在隔离性和效率二者之间寻找一个平衡点。

事务并发执行遇到的问题

脏读(Dirty Read): 脏读是指一个事务读到了另一个未提交事务修改过的数据。

如小王的账户中有100的余额,接下来有两个事务对小王的账户进行访问。

会话A会话B
begin;
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会话B
begin;
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会话B
begin;
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会话B
begin;
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会话B
begin;
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会话B
begin;
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事务隔离级别以及脏读、幻读、不可重复读的示例"这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注行业资讯频道!

事务 隔离 级别 数据 小王 客户 标准 账户 查询 余额 意味 服务器 语句 服务 不同 客户端 引擎 数据库 问题 小张 数据库的安全要保护哪些东西 数据库安全各自的含义是什么 生产安全数据库录入 数据库的安全性及管理 数据库安全策略包含哪些 海淀数据库安全审计系统 建立农村房屋安全信息数据库 易用的数据库客户端支持安全管理 连接数据库失败ssl安全错误 数据库的锁怎样保障安全 服务器安全软件免费版 数据库怎样创建 小火箭如何添加轻量云服务器节点 计算机网络技术三级大纲 济南博赛网络技术公司招聘 新华互联网科技系列之师资力量 网络安全hcie考试内容 石家庄标准软件开发服务咨询报价 软件开发的税务科目 江苏配件管理软件开发公司 黑客互联网科技 拔刀服务器 沈阳日佳网络技术怎么样 吉林网络技术咨询哪家好 如何开发pc软件开发 玉山县网络安全和信息化中心 电脑销售系统数据库使用说明书 广州臻淼网络技术有限公司 湖南智能土地资产管理软件开发 信息网络安全包括内容安全吗 网页上传服务器没效果 潍坊网络安全工程师培训 山特c6k服务器说明书 加强对网络安全的管理 吸引力软件开发 安徽名优网络技术服务项目 ns平台神之浩劫无法连接服务器 网络安全场景视频 微信的信息储存在服务器吗 网络安全个人要注意什么
0