Mysql加锁过程详解(2)-关于mysql 幻读理解
这里给出 mysql 幻读的比较形象的场景:
users: id 主键
1、T1:select * from users where id = 1;
2、T2:insert into `users`(`id`, `name`) values (1, 'big cat');
3、T1:insert into `users`(`id`, `name`) values (1, 'big cat');
T1 :主事务,检测表中是否有 id 为 1 的记录,没有则插入,这是我们期望的正常业务逻辑。
T2 :干扰事务,目的在于扰乱 T1 的正常的事务执行。
在 RR 隔离级别下,1、2 是会正常执行的,3 则会报错主键冲突,对于 T1 的业务来说是执行失败的,这里 T1 就是发生了幻读,因为T1读取的数据状态并不能支持他的下一步的业务,见鬼了一样。
在 Serializable 隔离级别下,1 执行时是会隐式的添加 gap 共享锁的,从而 2 会被阻塞,3 会正常执行,对于 T1 来说业务是正确的,成功的扼杀了扰乱业务的T2,对于T1来说他读取的状态是可以拿来支持业务的。
所以 mysql 的幻读并非什么读取两次返回结果集不同,而是事务在插入事先检测不存在的记录时,惊奇的发现这些数据已经存在了,之前的检测读获取到的数据如同鬼影一般。
这里要灵活的理解读取的意思,第一次select是读取,第二次的 insert 其实也属于隐式的读取,只不过是在 mysql 的机制中读取的,插入数据也是要先读取一下有没有主键冲突才能决定是否执行插入。
不可重复读侧重表达 读-读,幻读则是说 读-写,用写来证实读的是鬼影。
下面例子版本
SELECT VERSION();

例子1,读提交
|
a |
b |
|
SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED; |
|
|
SET AUTOCOMMIT=0; |
|
|
1.不可重复读 |
|
|
begin |
begin |
|
INSERT test VALUES(1,1); |
|
|
SELECT * FROM test;
|
SELECT * FROM test;
|
|
commit |
|
|
SELECT * FROM test; |
|
|
|
|
COMMIT |
|
|
B在一个事务的查询的结果变了,不可重复读 |
|
|
2.锁 |
|
|
begin |
begin |
|
INSERT test VALUES(2,2); |
|
|
SELECT * FROM test;
|
|
|
INSERT test VALUES(2,2); |
|
|
Lock wait timeout exceeded; try restarting transaction |
|
|
COMMIT |
COMMIT |
|
begin |
|
|
INSERT test VALUES(3,3); |
|
|
INSERT test VALUES(4,4); |
|
|
COMMIT |
|
|
BEGIN |
BEGIN |
|
SELECT COUNT(*) FROM test WHERE a>2; |
SELECT COUNT(*) FROM test WHERE a>2; |
|
INSERT test VALUES(5,5); |
|
|
SELECT COUNT(*) FROM test WHERE a>2; |
SELECT COUNT(*) FROM test WHERE a>2; |
|
COMMIT |
|
|
SELECT COUNT(*) FROM test WHERE a>2; |
SELECT COUNT(*) FROM test WHERE a>2; |
|
|
例子2:重复读
|
a |
b |
|
SET SESSION TRANSACTION ISOLATION LEVEL REPEATABLE READ; |
|
|
SET AUTOCOMMIT=0; |
|
|
1.可重复读 |
|
|
begin |
begin |
|
INSERT test VALUES(1,1); |
|
|
SELECT * FROM test;
|
SELECT * FROM test;
|
|
commit |
|
|
SELECT * FROM test; |
|
|
|
|
COMMIT |
|
|
BEGIN |
|
|
SELECT * FROM test; |
|
|
|
|
COMMIT |
|
|
B在一个事务的查询的没变 |
|
|
2锁 |
|
|
begin |
begin |
|
INSERT test VALUES(2,2); |
|
|
SELECT * FROM test;
|
|
|
INSERT test VALUES(2,2); |
|
|
Lock wait timeout exceeded; try restarting transaction |
|
|
COMMIT |
COMMIT |
|
3(幻读) |
|
|
BEGIN |
BEGIN |
|
INSERT test VALUES(3,3); |
|
|
SELECT * FROM test;
|
SELECT * FROM test;
|
|
COMMIT |
|
|
SELECT * FROM test;
|
|
|
INSERT test VALUES(3,3); |
|
|
Duplicate entry '3' for key 'PRIMARY' |
|
|
COMMIT |
|
|
BEGIN |
|
|
SELECT * FROM test;
|
|
|
COMMIT |
|
|
幻读,b明明查到没有,插入时候提示主键冲突,刚刚查询没有,出现幻觉? |
|
|
begin |
|
|
INSERT test VALUES(4,4); |
|
|
COMMIT |
|
|
4.可重复读 |
|
|
BEGIN |
BEGIN |
|
SELECT COUNT(*) FROM test WHERE a>2; |
SELECT COUNT(*) FROM test WHERE a>2; |
|
INSERT test VALUES(5,5); |
|
|
SELECT COUNT(*) FROM test WHERE a>2; |
SELECT COUNT(*) FROM test WHERE a>2; |
|
COMMIT |
|
|
BEGIN |
|
|
SELECT COUNT(*) FROM test WHERE a>2; |
SELECT COUNT(*) FROM test WHERE a>2; |
|
|
|
COMMIT |
COMMIT |
网上很多说范围啊,count等等都是不对的,不用于幻读
Mysql加锁过程详解(2)-关于mysql 幻读理解的更多相关文章
- Mysql加锁过程详解(8)-理解innodb的锁(record,gap,Next-Key lock)
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(9)-innodb下的记录锁,间隙锁,next-key锁
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(1)-基本知识
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(3)-关于mysql 幻读理解
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(4)-select for update/lock in share mode 对事务并发性影响
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(5)-innodb 多版本并发控制原理详解
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(6)-数据库隔离级别(1)
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(6)-数据库隔离级别(2)-通过例子理解事务的4种隔离级别
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
- Mysql加锁过程详解(7)-初步理解MySQL的gap锁
Mysql加锁过程详解(1)-基本知识 Mysql加锁过程详解(2)-关于mysql 幻读理解 Mysql加锁过程详解(3)-关于mysql 幻读理解 Mysql加锁过程详解(4)-select fo ...
随机推荐
- LOJ-10095(缩点的特殊使用)
题目链接:传送门 思路: 缩点求最值,但是有一点行不通,如果被选中的点才能缩点,否则缩点没有意义: 所以就先缩选中的点,然后从小到大统计没有缩点的点,就是NO: 如果找最小值,就是一个环里的最小值,然 ...
- vuex的mutations传值
mutations是要通过方法触发的,用于更改store里的数据的.this.$store.commit("mutationsName") 例子: store.js import ...
- vim配置文件.vimrc
20171127备份 syntax on "自动语法高亮 set number "显示行号 set autoindent "回车后自动缩进 set tabstop=4 & ...
- 简单的JavaScript图像延迟加载库Echo.js
插件描述:和 Lazy Load 一样,Echo.js 也是一个用于图像延迟加载 JavaScript.不同的是 Lazy Load 是基于 jQuery 的插件,而 Echo.js 不依赖于 jQu ...
- 在原生Windows安装Keras
既然要深入学习,就不能和时代脱节,所以选择了keras,资源相对比较丰富.由于Windows饱受歧视,各种文档都不推荐使用.但我又没有换系统的成本,所以还是凑合下,毕竟他们给出了方法,稍微折腾一下还是 ...
- 【repost】Python正则表达式
星光海豚 python正则表达式详解 正则表达式是一个很强大的字符串处理工具,几乎任何关于字符串的操作都可以使用正则表达式来完成,作为一个爬虫工作者,每天和字符串打交道,正则表达式更是不可或缺的技 ...
- Mybatis in 查询
1.先创建一个传参的工具类 import java.util.HashMap; /** * * ClassName: DataMap * @Description: 封装Map, * @date 20 ...
- es6的几种写法
语法:1.只有一个参数,可以不用写小括号: var single = a => a; //相当于var single = function(a){return a;}console.log(si ...
- VSCode插件开发全攻略(七)WebView
更多文章请戳VSCode插件开发全攻略系列目录导航. 什么是Webview 大家都知道,整个VSCode编辑器就是一张大的网页,其实,我们还可以在Visual Studio Code中创建完全自定义的 ...
- 为什么使用 Spring Boot?
Spring 是一个非常流行的基于Java语言的开发框架,此框架用来构建web和企业应用程序.与许多其他仅关注一个领域的框架不同,Spring框架提供了广泛的功能,通过其组合项目满足现代业务需求. S ...

















