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 ...
随机推荐
- Cortext-A7_i.MX 6ULL——多模式DDR控制器(MMDC)
1.概述 i.MX 6ULL系列芯片的MMDC是一个多模式DDR控制器,支持DDR3/DDR3Lx16和LPDDR2x16的存储类型,MMDC是可配置,高性能,优化的内存控制器. 注:DDR3/DDR ...
- Kinfu配置指南
Kinfu配置指南 欢迎加入Kinfu讨论群:563741937 写给准备配置pcl-kinfu的同学,这个是我的配置方法,已经能用的请无视. 我的配置:Windows7,VS2010(用这个的是不是 ...
- 京东Alpha平台开发笔记系列(二)
第一篇博文简单讲了一下京东Alpha平台与个人idea技能,本篇将讲解Alpha平台与个人开发需要的一些知识,下面开篇 ——>>> 上图就是京东Alpha技能平台的首页,Skill平 ...
- java面试一、1.4锁机制
免责声明: 本文内容多来自网络文章,转载为个人收藏,分享知识,如有侵权,请联系博主进行删除. 1.3.锁机制 说说线程安全问题,什么是线程安全,如何保证线程安全 线程安全:当多个线程访问某一个 ...
- mybatis注解SQL
在网上找了很久,特别是批量插入,很久都没有找到,终于最后一不小心就搞出来了.所以想写个随笔保存下来,一方面想提高自己的总结能力,一方面为了结识有相同兴趣的朋友(第一篇博客我的天纳
- linux vsftp 简单配置
查看自己是否安装vsftp rpm -qa | grep vsftp rpm -qa 查看自己已安装的包 过滤vsftp systemctl rsetart vsftpd 重启服务 先关闭防火墙 sy ...
- zabbix环境搭建
zabbix介绍 zabbix是一个开源的监控软件集成了nagos和cat的优势 而且有很多自带的插件可以使用,而且还有api接口供我们使用 zabbix还支持自定义监控项 初始环境- centos ...
- ElasticSearch是如何实现分布式的?
面试题 es 的分布式架构原理能说一下么(es 是如何实现分布式的啊)? 面试官心理分析 在搜索这块,lucene 是最流行的搜索库.几年前业内一般都问,你了解 lucene 吗?你知道倒排索引的原理 ...
- Windows下Tomcat内存占用过高问题跟踪(ProcessExplorer+jstack)
一.问题描述 Tomcat下面部署很多个java项目的war包,tomcat启动一段时间后,发现cpu占用过高,整个界面卡死! 二.通过process explorer查看java进程下的线程 pro ...
- Docker学习笔记-Redis 安装
拉取官方的镜像 docker pull redis:3.2 查看 docker images redis 运行容器 docker run -p 6379:6379 -v $PWD/data:/data ...