聊聊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.事务的标准特征 一个运行良好的事务处理系统,必须具备原子性.一致 ...
随机推荐
- 学军中学第三届“图灵杯”趣味网络邀请赛——中级T4.欧拉回路 (图论,哈希)
题面 补题链接 7 5 6 7 1 2 3 3 13 5 30 50 10 30 70 8 题解 存在欧拉回路的条件是:1. 每个点的度数都是偶数.2. 有边的连通块最多一个. 数据范围是允许我们 n ...
- python随机值生成的常用方法
一.随机整数1.包含上下限:[a, b] import random #1.随机整数:包含上下限:[a, b] for i in range(10): print(random.randint(0,5 ...
- Servlet小结
1.sevlet的生命周期 用户在发送第一次请求的时候Servlet对象被实例化(AServlet的构造方法被执行了.并且执行的是无参数构造方法.) AServlet对象被创建出来之后,Tomcat服 ...
- 生信云实证Vol.12:王者带飞LeDock!开箱即用&一键定位分子库+全流程自动化,3.5小时完成20万分子对接
LeDock是苏黎世大学Zhao HongTao在博士期间开发的一款分子对接软件,专为快速准确地将小分子灵活对接到蛋白质而设计.LeDock优于大部分商业软件,在Astex多样性集合上实现了大于90% ...
- TDengine概述以及架构模型
TDengine TDengine是一个高效的存储.查询.分析时序大数据的平台,专为物联网.车联网.工业互联网.运维监测等优化而设计. 您可以像使用关系型数据库MySQL一样来使用它. TDengin ...
- 【项目实战】自备相机+IMU跑通Vins-Mono记录
前言 初次接触SLAM,公司要求用自己的设备来跑通vinsmono这个程序,虽然已经跑通了别人的数据包,但是真正自己上手来运行这个程序,发现真的是困难重重,特意在此记载下来整个过程,以供大家参考. 我 ...
- innodb_flush_log_at_trx_commit 和 sync_binlog 参数详 解
"innodb_flush_log_at_trx_commit"和"sync_binlog"两个参数是控制 MySQL 磁盘写入策略以及数据安全性的关键参数.当 ...
- ECON 模式
ECON模式通过调节发动机和空调系统的性能,有效提高燃油经济性. 在D行驶档的时候开启
- windows系统下使用bat脚本文件设置 JDK 系统环境变量
号开头的行不要写在bat文件中 # java_init.bat # 注意文件换行符是windows系统下的(CR LF),文件编码是ANSI # path变量追加这个可以拓展到tomcat,mysql ...
- 使用growpart扩容CentOS虚拟机磁盘
说明 一块磁盘,比如有130G,但是只有100G拿来用了,安装俩分区,一个是swap,另一个是根分区/,根分区采用的是LVM逻辑卷管理. 现在要做的是把剩余的这空闲30G分配给根分区使用 这个并不是新 ...