【MySQL】可重复读模式下 unique key失效案例
今天上午文能提笔安天下,武能上马定乾坤的登博给团队出了一道题目,谁先复现问题,奖励星巴克一杯。激起了一群忙碌的屌丝DBA的极大热情。问题是这样滴,如下图
登博提示了几个细节:
1. code上的uk并未失效。
2. rr隔离级别。
3. 有并发线程的操作。
二 【原理分析】
1 事务隔离级别的基础知识:
- 未提交读(Read Uncommitted):允许脏读,也就是可能读取到其他会话中未提交事务修改的数据。
- 提交读(Read Committed):只能读取到已经提交的数据。Oracle等多数数据库默认都是该级别 (不重复读)。
- 可重复读(Repeated Read):可重复读。在同一个事务内的查询都是事务开始时刻一致的,InnoDB默认级别。在SQL标准中,该隔离级别消除了不可重复读,但是还存在幻象读。
- 串行读(Serializable):完全串行化的读,每次读都需要获得表级共享锁,读写相互都会阻塞。
这里重点说一下RR 模式:可重复读 是指在一个事务内,多次读同一数据。在这个事务还没有结束时,另外一个事务也访问该同一数据。那么,在第一个事务中的两次读数据之间,即使第二个事务对数据进行修改,第一个事务两次读到的的数据是一样的。这样就发生了在一个事务内两次读到的数据是一样的,因此称为是可重复读。
2 MVCC 的读操作
在MVCC并发控制中,读操作可以分成两类:快照读 (snapshot read)与当前读 (current read)。快照读,读取的是记录的可见版本 (有可能是历史版本),不用加锁。
当前读,读取的是记录的最新版本,并且,当前读返回的记录,都会加上锁,保证其他事务不会再并发修改这条记录。
快照读:简单的select操作,属于快照读,不加锁。
select * from table where ?;
当前读:特殊的读操作,插入/更新/删除操作,属于当前读,需要加锁。
select * from table where ? lock in share mode;
select * from table where ? for update;
insert into table values (…);
update table set ? where ?;
delete from table where ?;
所有以上的语句,都属于当前读,读取记录的最新版本。并且,读取之后,还需要保证其他并发事务不能修改当前记录,对读取记录加锁。其中,除了第一条语句,对读取记录加S锁 (共享锁)外,其他的操作,都加的是X锁 (排它锁)。
注意:insert操作可能会触发Unique Key的冲突检查,也会进行一个当前读。
三【解决.复现】
测试版本: 5.5.18 5.6.16 均可复现。
现在我们根据上述理论信息进行复现问题,具体的实现步骤如下:
注意 数据库的隔离级别为RR
session 1 | session 2 |
root@test 08:47:41>set global tx_isolation='REPEATABLE-READ'; Query OK, 0 rows affected (0.00 sec) root@test 08:53:16>set autocommit=0; Query OK, 0 rows affected (0.00 sec) root@test 08:53:22>insert into yy values(1,20,13); Query OK, 1 row affected (0.00 sec) root@test 08:53:31>commit; Query OK, 0 rows affected (0.00 sec) root@test 08:53:39>select * from yy; +----+------+------+ | id | code | val | +----+------+------+ | 1 | 20 | 13 | +----+------+------+ 1 row in set (0.00 sec) |
|
root@test 08:53:46>select * from yy; +----+------+------+ | id | code | val | +----+------+------+ | 1 | 20 | 13 | +----+------+------+ 1 row in set (0.00 sec) root@test 08:53:53>delete from yy where id=1; Query OK, 1 row affected (0.00 sec) root@test 08:53:59>commit; |
|
root@test 08:54:10>insert into yy values(2,20,13); Query OK, 1 row affected (5.59 sec) root@test 08:54:23>select * from yy; +----+------+------+ | id | code | val | +----+------+------+ | 1 | 20 | 13 | | 2 | 20 | 13 | +----+------+------+ 2 rows in set (0.00 sec) |
当session 2中将id=1 的删除之后,session1 进行insert操作时,触发unique key冲突检查,此时因为id=1 code=20的数据已经被物理删除了,MySQL 检查无冲突,进行insert insert into yy values(2,20,13); 便成功了。
四【结果展示】
五【参考资料】
1 《MySQL 加锁分析》
2 《Innodb中的事务隔离级别和锁的关系》
转载于:https://www.cnblogs.com/cyt1153/p/6576009.html
【MySQL】可重复读模式下 unique key失效案例相关推荐
- mysql 可重复读 原理,Mysql可重复读原理
概念 可重复读的实现 Repeatable Read(可重复读):一个事务在执行过程中可以看到其他事务已经提交的新插入的记录(读已经提交的,其实是读早于本事务开始且已经提交的),但是不能看到其他事务对 ...
- Mysql可重复读隔离级别下如何解决幻读
Mysql可重复读隔离级别下如何解决幻读 一些概念 具体加锁说明 使用主键索引进行等值查询 使用主键索引进行范围查询 使用二级索引进行等值查询 使用二级索引进行范围查询 一些概念 幻读:在一次事务中, ...
- MySQL 可重复读隔离级别与幻读
在MySQL可重复读的隔离级别下,能很大程度上避免幻读,而不能完全避免. 场景复现 环境信息: MySQL版本:5.7.23-log 隔离级别:REPEATABLE-READ 测试数据: SET NA ...
- MySQL可重复读级别能够解决幻读吗
引言 之前在深入了解数据库理论的时候,了解到事物的不同隔离级别可能存在的问题.为了更好的理解所以在MySQL数据库中测试复现这些问题.关于脏读和不可重复读在相应的隔离级别下都很容易的复现了.但是对于幻 ...
- MySQL可重复读隔离级别为何没有解决幻读(MVCC原理简介)
MySQL可重复读隔离级别为何没有解决幻读(MVCC原理简介) 一.MCVV简介 二.可重复读隔离级别能解决幻读? 三.什么是当前读和快照读? 四.MVCC的实现原理 五.RC,RR级别下的InnoD ...
- mysql可重复读和幻读的理解
mysql可重复读和幻读的理解 可重复读和幻读的定义 最后总结 参考资料 很多教程和书籍对mysql的可重复读和幻读的解释都比较含糊,本文结合原理和其他的考证,深入分析下. 这里讨论的引擎是常用的In ...
- MySQL可重复读和读已提交实现原理,深入理解MVCC。
1.隔离级别 MySQL中隔离级别分为4种,提未交读.读已提交.可重复读.串行化.同时MySQL默认隔离级别为可重复读. 图片 查看MySQL隔离级别 SELECT @@tx_isolation 设置 ...
- MySQL可重复读-问题实践
MySQL可重复读之幻读问题 MySQL事务存储引擎InnoDB的默认隔离级别为可重复读,在该隔离级别下,可以很大程度上避免幻读的问题,完全避免脏读和不可重复读问题,接下来通过实际的测试看看这些场景是 ...
- mysql 可重复读 快照_MYSQL可重复读及原理、快照读和当前读
什么是可重复读 可重复读的实现 Repeatable Read(可重复读):一个事务在执行过程中可以看到其他事务已经提交的新插入的记录(读已经提交的,其实是读早于本事务开始且已经提交的),但是不能看到 ...
最新文章
- python简单单元测试示范卷_Python 单元测试的简单示例
- Facebook收购GrokStyle:布局AI零售
- MUI+H5手机上传照片 支持多图片上传和拍照上传
- jQuery 效果 - animate() 方法
- 茅台防伪溯源服务器临时维护,如何使用茅台防伪溯源系统?能辨别茅台酒真假?...
- 安卓桌布显示的dip和px
- Anaconda 完全入门指南
- Netrunner 2019.04 Rolling 版本发布
- Oracle sqlldr 在DOS窗口导入多列数据到数据库表
- 流量 起伏大_广西崇左的德天跨国大瀑布,一半是中国一半是越南,天下奇观!...
- 诺基亚造平板到底前途几何?
- Dcmtk在PACS开发中的应用(基础篇) 第二章 打印影像(胶片) 作者 冷家锋
- AndroidStudio_android蓝牙开发总结_连接硬件设备_测量_血压_血糖_握力_心电_等---Android原生开发工作笔记244
- SetLayeredWindowAttributes 设置窗口透明
- SQL Server 2008 报表服务入门
- python画圆形螺旋线_中秋节到了,送你一个Python做的Crossin牌“月饼”
- 吃完饭打嗝原因及治疗方法(分享)
- EXCEL VBA编程基础
- 路由器450m和1200m有什么区别
- 智能工厂数据采集方案
热门文章
- 四、物理优化(2)索引视图
- 初学Windows编程笔记1——窗口和消息
- 1.9 函数-C++编程模块
- php服务器怎么返回信息,php怎么返回服务器ip地址
- 原生 AJAX的相关介绍
- Thymeleaf读取model里面的对象||Thymeleaf读取model里面的集合||Themeleaf在js中取值||访问带参数的消息||ThymeleafObjects的使用[取三大作域的值
- Servlet 的常见错误总结
- jQuery取得select选择的文本与值
- JavaScript toFixed(num) 函数
- PyQt5 图形界面 - 配置界面跟随窗口大小调整灵活伸缩,设置页面控件居中显示实例演示