聊聊mysql的事务
今天来聊聊事务的四大特性以及其实现原理,需结合之前写的mysql是如何实现mvcc的来理解,因为大多数的实现都是基于mvcc的,理论介绍完后会通过实例来演示mvcc又是如何实现这些隔离级别的
事务的四大特性
- 原子性
事务的执行要么全部成功要么全部失败,成功的时候直接提交就好,某些环节失败的时候是需要回滚的,这就要用到回滚日志(undolog)根据当前记录的回滚指针去undolog定位历史记录了; - 一致性
事务的其他特性就是为了保证一致性的 - 隔离性
两个事务之间的执行应该是隔离开的,各自造成的数据改变不会相互影响,有四个隔离级别下面会一一解释 - 持久性
事务提交后持久化到硬盘,通过redo log实现
隔离级别
说明:下面的概念解释假设有两个事务A、B处于活跃状态
- 读未提交(READ-UNCOMMITTED)
事务B读到了事务A新增但未提交的数据;这种事务未提交就被其他事务读取到的现象叫做脏读;而在实际生产中这是毫无意义的,所以几乎不会被用到 - 读已提交(READ-UNCOMMITTED)
是oracle的默认隔离级别;事务B读到了事务A修改了并且已经提交了的数据;分两种情况,一是事务A修改了数据导致B在第二次读的时候与第一次读到的结果不一样,这种现象叫做不可重复读;二是事务A新增了数据导致B在第二次查询时候与第一次查询时读到的记录数不一致,这种现状叫做幻读;即RC级别下可能会出现幻读和不可重复读的现象 - 可重复读(REPEATABLE-READ)
是mysql的默认隔离级别;事务A对数据的操作不影响事务B,即事务B在事务A修改数据前后查的数据是一致的;这种级别看似不会出现问题,但是当当前读和快照读混用的时候也是会出现幻读或者不可重复读的问题的,下面我会演示这一现象;而mysql是通过mvcc+间隙锁来解决这一问题的 - 串行(SERIALIZABLE)
事务串行执行,效率低但不会出现脏读、幻读、不可重复读的现象
小总结:综上所述,幻读可以理解成相同条件下事务前后读取到的记录数不一样,而不可重复读可理解成事务前后读取都的内容不一样,记录数可以一样;
不同隔离级别下的实操
数据准备
- 创建一个测试表test
CREATE TABLE test (
id int NOT NULL AUTO_INCREMENT,
name varchar(255) DEFAULT NULL,
age int DEFAULT NULL,
PRIMARY KEY (id)
)
- 插入原始测试数据
insert into test values(10,'zhangsan',19);
insert into test values(20,'lisi',21);
insert into test values(30,'wangwu',21);
可重复读(RR)隔离级别下测试
1.打开两个命令行窗口并关闭事务的自动提交;然后分别在两个窗口手动开启事务;如图:

2.在A事务窗口执行查询操作:select * from test where age = 21;结果和预期一致符合条件的是两条数据,如图:

3.在B事务窗口执行插入操作insert into test values(40,'zhaoliu',21);后提交事务并进行查询select * from test where age = 21;此时查询到的是三条记录,如图:

4.在事务A窗口重新执行select * from test where age = 21;发现得到的结果与第一次一致还是两条

分析:
综上可以发现,事务A在事务周期内在事务B对表做了新增数据的情况下两次查询结果仍然一致,没有出现幻读、不可重复读的情况,是因为mysql的默认隔离级别是RR(可重复读),而正常情况下RR是可以避免幻读和不可重复读的情况的,除非在事务周期内混用了当前读和快照读,下面来演示RR级别下出现幻读的情况;
- 5.在上面的基础上我们在事务A的查询语句基础上新增一个for update语句:select * from test where age = 21 for update;此时我们就会发现记录数变成了3条(如图),即出现了幻读的情况,因为类似for update、delete等语句是会使用当前读的,而之前的查询语句用的是快照读,这就出现了快照读和当前读混用的现象从而出现了幻读

读已提交(RC)隔离界别下的测试
步骤与上面几乎一致,只是需要在开启事务之前将隔离级别设置成读已提交(RC),如图:

接下来重复上面的步骤:
- 1.关闭事务的自动提交;手动开启两个事务,这里就不上图了
- 2.事务A执行select * from test where age = 21;得到的结果是三条(上面的测试新增了一条数据),如图:

- 3.事务B执行insert into test values(50,'qianqi',21);后提交事务并进行查询select * from test where age = 21;此时查询到的是四条记录,如图:

- 4.在事务A窗口重新执行select * from test where age = 21;此时我们发现得到结果却是4条记录与第一次查询的结果不一致,这就出现了幻读的现象,如图:

思考:为什么在测试步骤和数据不变的情况下不同的隔离级别下查询的结果不一致呢?
这是因为两种隔离级别下生成readView的时机不一样,mysql默认的可重复读级别下是在事务中的第一次查询时生成的readview,后续在没有当前读的情况下事务内所有的查询都是基于第一次查询生成的readview的快照读;而读已提交的隔离级别下在事务内的每次查询都会生成一个readview,所以造成了两种不同的现象;
聊聊mysql的事务的更多相关文章
- 数据库事务系列-MySQL跨行事务模型
说来和MySQL倒是有缘,毕业的第一份工作就被分配到了RDS团队,主要负责把MySQL弄到云上做成数据库服务.虽说整天和MySQL打交道,但说实话那段时间并没有很深入的理解MySQL内核,做的事情基本 ...
- 聊聊MySQL的加锁规则《死磕MySQL系列 十五》
大家好,我是咔咔 不期速成,日拱一卒 本期来聊聊MySQL的加锁规则,知道这些规则后可以判断SQL语句的加锁范围,同时也可以写出更好的SQL语句,防止幻读问题的产生,在能力范围内最大程度的提升MySQ ...
- MySQL 数据库事务与复制
好久没有写技术文章了,因为一直在思考 「后端分布式」这个系列到底怎么写才合适. 最近基本想清楚了,「后端分布式」包括「分布式存储」和 「分布式计算」两大类. 结合实际工作中碰到的问题,以寻找答案的方式 ...
- Mysql分布式事务
关于Mysql分布式事务介绍,可参考:http://blog.csdn.net/luckyjiuyi/article/details/46955337 分为两个阶段:准备和执行阶段.有两个角色:事务的 ...
- mysql的事务和select...for update
一.mysql的事务mysql的事务有两种方式:1.SET AUTOCOMMIT=0;也就是关闭了自动提交,那么任何commit或rollback语句都可以触发事务提交;如果SET AUTOCOMMI ...
- MySQL的事务
MySQL的事务 1.事务:事务是由一步或者几步数据库操作序列组成的逻辑执行单元,这一系列操作要么全部执行,要么全部放弃执行. 2.事务具备的四个特性(简称为ACID性): (1)原子性(Atomic ...
- mysql 分布式事务
php + mysql 分布式事务 事务(Transaction)是访问并可能更新数据库中各种数据项的一个程序执行单元: 事务应该具有4个属性:原子性.一致性.隔离性.持续性 原子性(atomicit ...
- MySQL之事务隔离级别--转载
转自:http://793404905.blog.51cto.com/6179428/1615550 本文通过实例展示MySQL事务的四种隔离级别. 1 概念阐述 1)Read Uncommitted ...
- MySQL数据库事务剖析
MySQL数据库事务剖析 事务就是一组原子性的SQL查询,是一个独立的执行单元.事务内的语句,要么全部执行成功,要么全部执行失败. 1.事务的标准特征 一个运行良好的事务处理系统,必须具备原子性.一致 ...
随机推荐
- AI 音辨世界:艺术小白的我,靠这个AI模型,速识音乐流派选择音乐 ⛵
作者:韩信子@ShowMeAI 数据分析实战系列:https://www.showmeai.tech/tutorials/40 机器学习实战系列:https://www.showmeai.tech/t ...
- CF1204E Natasha, Sasha and the Prefix Sums (卡塔兰数推理)
题面 题解 把题意变换一下,从(0,0)走到(n,m),每次只能网右或往上走,所以假设最大前缀和为f(n),那么走的时候就要到达但不超过 y = x-f(n) 这条线, 我们可以枚举答案,然后乘上方案 ...
- LibreCAD常用命令
目录 常见命令 常见命令 .text_center { text-align: center } \3cp>.text_left { } 动作命令 命令 绘制直线 相对坐标系 @长度<角度 ...
- centos7部署k8s(1master1node)
〇.前言 就想多学学罢了 准备环境: centos7 master 8GB 172.26.130.204 centos7 node 8GB 172.26.130.205 yum源就阿里源就好... 一 ...
- 二进制redis集群部署
二进制redis集群部署 〇.前言 无聊想学罢了 准备环境: 三台centos7 1C1GB即可 三个路相连的地址 主机 IP 节点-角色-实例(端口) redis1 172.16.106.128 M ...
- 华南理工大学 Python第7章课后小测-2
1.(单选)以下选项中使Python脚本程序转变为可执行程序的第三方库的是(本题分数:3)A) NetworkxB) pyinstallC) RequestsD) PyPDF2您的答案:B 正确率: ...
- Elasticsearch: 使用URI Search
在Elasticsearch中,我们可以使用_search终端进行搜索.这个在我之前的文章 "开始使用Elasticsearch (2)" 中有很多的描述.针对这种搜索,我们可以使 ...
- ES小知识点
elasticsearch.yml配置文件 network.host: _site_ # network.host设置为"_site_",表明它绑定到我们的本地电脑的IP地址 di ...
- Maven+SpringMVC+Dubbo 简单的入门demo配置
转载自:https://cloud.tencent.com/developer/article/1010636 之前一直听说dubbo,是一个很厉害的分布式服务框架,而且巴巴将其开源,这对于咱们广大程 ...
- 打印 Logger 日志时,需不需要再封装一下工具类?
在开发过程中,打印日志是必不可少的,因为日志关乎于应用的问题排查.应用监控等.现在打印日志一般都是使用 slf4j,因为使用日志门面,有助于打印方式统一,即使后面更换日志框架,也非常方便.在 < ...