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事务隔离级别以及脏读、幻读、不可重复读示例相关推荐
- MySQL --- 19♪ 进阶15 TCL事务控制语言--建立结束事务/设置断点--默认隔离级别--脏读/幻读/不可重复读
#TCL事物控制语言 : /* Transaction control language : 事物控制语言 事务: 一个或者一组sql语句组成一个执行单元,这个执行单元要么全部执行,要 ...
- mysql 中的脏读与幻读_一文带你理解脏读,幻读,不可重复读与mysql的锁,事务隔离机制...
首先说一下数据库事务的四大特性 1 ACID 事务的四大特性是ACID(不是"酸"....) (1) A:原子性(Atomicity) 原子性指的是事务要么完全执行,要么完全不执行 ...
- mysql 可重复读 悲观锁_一文带你理解脏读,幻读,不可重复读与mysql的锁,事务隔离机制...
首先说一下数据库事务的四大特性 1 ACID 事务的四大特性是ACID(不是"酸"....) (1) A:原子性(Atomicity) 原子性指的是事务要么完全执行,要么完全不执行 ...
- mysql悲观锁会有脏数据吗_一文带你理解脏读,幻读,不可重复读与mysql的锁,事务隔离机制...
首先说一下数据库事务的四大特性 1 ACID 事务的四大特性是ACID(不是"酸"....) (1) A:原子性(Atomicity) 原子性指的是事务要么完全执行,要么完全不执行 ...
- 事务的4种隔离级别,3大问题,脏读幻读不可重复读
一.数据库事务隔离级别 数据库事务的隔离级别有4个,由低到高依次为Read uncommitted .Read committed .Repeatable read .Serializable ,这四 ...
- 数据库几个事务相关的知识点(脏读幻读不可重复读以及如何避免)
0. 设置MYSQL的隔离界别 # 查询数据库当前事务隔离级别 mysql> select @@global.transaction_isolation,@@transaction_isolat ...
- 脏读 幻读 不可重复读 及其解决方法
一个事务读到另外一个事务还没有提交的数据,我们称之为脏读.解决方法:把事务隔离级别调整到READ COMMITTED 一个事务先后读取同一条记录,但两次读取的数据不同,我们称之为不可重复读.解决方法: ...
- 脏读幻读不可重复读的区别
1.脏读:脏读就是指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据. 例如: 张三的工资为5000,事务A中把他的工 ...
- mysql 默认事务隔离级别_一文读懂MySQL的事务隔离级别及MVCC机制
回顾前文: <一文学会MySQL的explain工具> <一文读懂MySQL的索引结构及查询优化> (同时再次强调,这几篇关于MySQL的探究都是基于5.7版本,相关总结与结论 ...
最新文章
- MySQL高性能优化实战总结!
- android的NDK和java进行本地socket通信
- 【B站视频教程笔记】基于VSCode和CMake实现C/C++开发 | Linux篇(gcc/g++)(安装、配置、使用详细教程)(VSCode教程)(CMake教程)(精!)
- 杭电1003 java_杭电ACM1003题怎么理解?
- mysql新建用户并授权_Mysql中新建用户及授权的方法分享
- jquery validate 笔记
- GIMP的Path的import和export
- Day001 20210206
- Godot GUI探讨
- R数据分析:混合效应模型的可视化解释,再不懂就真没办法
- 自我介绍及阅读与思考及未来的期望
- 励志:滴滴打车App初期是怎么推广的?
- 估计很多人不知道:在PowerPoint中插入图片的三种方式用法和解析
- ROS讲座 关于ROS2和Gazebo C++ in Open Source Robotics
- 如果360与QQ冲突下去不兼容,你会放弃哪一个?
- ransomware(假的勒索病毒)逆向分析
- kzzi k980 三模键盘 说明书
- Helm 创建一个NOTES.txt文件
- scala-尾递归,Array.newbuilder,二维数组
- 基于MATLAB的疲劳检测系统