注:begin或start transaction并不是一个事务的起点,而是在执行它们之后的第一个操作InnoDB表的语句,事务才真正开始。start transaction with consistent snapshot命令可以马上启动一个事务。

1、隔离级别

1.1、基本概念

读未提交

当前事务能读取到其他事务尚未提交的修改值。

读提交

当前事务能读取到其他事务已经提交的修改值。

可重复读

一个事务在任何时刻查询到的结果跟事务一开始启动时查询的结果一样。

串行化

强行使所有事务串行执行。

1.2、小测试一

理解了隔离级别的基本概念后,我们做个小测试,看看是不是真的理解了,下面两个会话按顺序执行,最后在各个隔离级别下的V1、V2值各为多少呢?

sessionA sessionB
启动事务
查询得到值1
启动事务
把值1更新成值2
查询得到值V1
提交事务
查询得到值V2
提交事务
查询得到值V3

读未提交可以读取到其他事务未提交的值,所以V1读取到了sessionB未提交的更改,值为2。
读提交不能读取到其他事务未提交的值,所以V1的值为1,当查询V2时,sessionB已经提交了,这时读取到的V2为2。
可重复读查询得到的值为事务启动时的值,所以V1和V2值都为1。
串行化时,sessionA启动事务后,sessionB会被卡住,直至sessionA的事务提交后才执行,所以V1和V2都是1,V3查询时sessionB已经提交事务,所以V3的值为2。

V1 V2 V3
读未提交 2 2 2
读提交 1 2 2
可重复读 1 1 2
串行化 1 1 2

1.3、小测试二

上面的小测试一加深了我们对事务隔离级别的了解,下面我们针对可重复读隔离级别下对小测试一进行修改再加深理解。例子如下:

sessionA sessionB
启动事务
查询得V值为1
启动事务
set V=V+1
查询得到值V1
提交事务
set V=V+1
查询得到值V2
提交事务
查询得到值V3

小测试一可重复读隔离级别下V1、V2的值都为1,V3的值为2。根据小测试一的思路,你可能会得出小测试二的结果为V1为1,V2为2,那么V3是多少呢?
其实正确的答案是V1为1,V2为3,V3为3。如果sessionA中的set V=V+1是set V=1+1的话,这样就丢失了sessionB更新的操作,使数据不正确,这时因为sessionB提交了,所以sessionA中的更新同一个值为当前读,为set V=2+1,所以V2为3。小测试一中读取的数据在概念中为一致性读
如果流程中把sessionB的提交事务放在sessionA的提交事务后面呢?这个时候sessionA的set V=V+1会被阻塞,直至sessionB的事务提交后才会执行,结果还是V1为1,V2为3,V3为3。

MySQL的事务隔离级别默认为可重复读,oracle的事务隔离级别默认为读提交,所以在把oracle迁移至MySQL时需要记得把MySQL的设置读提交后操作。
可重复读在事务启动时会创建一个视图,后继查询都在此视图上操作。
读提交在事务启动时也会创建一个视图,但其他事务更新操作会直接更新此视图。
读未提交直接读取数据库中最新的数据。
串行化直接进行加锁避免其他事务修改。

view1
view2
1 2

如上表格所示视图,行数据从1更新为2的过程对应视图view1到view2的过程。
读提交隔离级别下,事务启动时是view1视图,当sessionB发生更新数据后,视图view1会更新成视图view2,后续查询在view2上进行。
可重复读隔离级别下,事务启动时是view1视图,当sessionB发生更新数据后,事务还是照样读取view1视图。
可重复读隔离级别下,是怎么找到view1视图的呢?
其实这就是MVCC的功能了,事务在启动时会获得系统会分配一个按顺序递增的transaction id,然后事务在更新数据时会把这个transaction id 分配给这一行数据当做trx row_id,顺便记录undo log日志。比如上面sessionA事务在启动时拿到的transaction id为889,sessionB的transaction id为890,那么sessionB更新完后数据的trx row_id为890。在查询V2的值时,sessionA发现数据版本为890,当前自己事务id为889,所以利用undo log“回滚”显示出上一版本的数据,上一版本可以是889也可能是比889小的trx row_id,取值。

1.4、隔离级别解决脏读、不可重复读、幻读

声明一点,读未提交、读提交、可重复读、串行化隔离级别依次递增,隔离得越严实,效率就越低。
脏读

事务读取到其他事务修改后未提交的值。

不可重复读

事务在启动时读取到的值,跟事务在执行过程中重新读取到的值不一样。

幻读

事务启动后,其他事务对数据库新插入的行被本事务读取到了。幻读仅专指新插入的行

根据上述脏读、不可重复读、幻读的概念与隔离级别能做到的,我可以得出以下关系:

脏读 不可重复读 幻读
读未提交 可能 可能 可能
读提交 不可能 可能 可能
可重复读 不可能 不可能 可能
串行化 不可能 不可能 不可能

2、ACID

A.Atomicity.原子性

一个事务的操作是要么全部成功,要么全部失败。例如A给B转账100块,保证A账户会减少100块,B账户增加100块,这100块是A转过来的,不出现A账户减少了,B账户却没对应增加100块的情况。

原子性通过undo log实现,undo log记录了回滚信息,当事务失败或进行rollback时,就会根据信息回滚到操作前的样子。
当delete时,undo log会相应记录被删除数据的信息,当发生回滚时,进行insert操作。
当update时,undo log会记录更新操作的信息,当发生回滚时,进行update操作。
当insert时,undo log会记录对应的主键,当发生回滚时,进行delete操作。

C.Consistency.一致性

事务在执行前后,数据从一种合法的状态,转变成另一种合法的状态。怎么样才算合法,都是可以自己定义的,比如事务启动时A用户账户了1000块,B用户有500块,这个时候是启动时合法的状态。A用户向B用户转账100块,A用户账户余额变成900,B变成600,这是结束时合法的状态。或者平台会扣除1%的手续费,转账完成后A用户余额为899,B用户变成600,平台账户增加1块钱,这也是合法的状态。

一致性是通过原子性、隔离性、持久性来保证的。所以只有保证了AID特性,才能保证C特性。

I.Isolation.隔离性

一个事务的查询结果不会被其他事务所影响,事务之间是隔离的。例如X事务在启动时查询id=1的结果为1,X事务在没有对此记录进行修改操作的时候,多次查询结果还是为1,不管其他事务是否对此记录进行了修改。

隔离性通过MVCC实现,具体实现可以参考上面中的小测试逻辑。

D.Durability.持久性

一个事务执行完成了,所修改的数据能保存至数据库,不管宕机还是突然断电,数据库重新启动后查询到的数据为事务操作过后的数据。

持久性通过redo log实现,MySQL中有个WAL的逻辑,就是编辑数据时,会先更新redo log日志,再更新内存,最后再写磁盘。写redo log日志是一个非常快的操作,而更新完内存后,写进磁盘是很忙的。当还没写进磁盘时发生宕机等,MySQL重启后会自动去对比redo log日志与磁盘内容,把未刷进磁盘的数据刷进磁盘中,从而保证了数据的持久性。

3、扩展知识

undo log

回滚日志记录数据修改的逆向还原操作记录。例如数据值从1按顺序修改成了2、3、4。那么回滚日志就会按顺序记录:2->1,3->2,4->3

MVCC

Multi-Version Concurrency Control是MySQL数据库中的多版本并发控制,就是同一条记录在数据库中是可以存在多个版本。依靠undo log与乐观锁实现。

乐观锁

乐观地认为不会冲突,在最后需要修改数据的时候去拿一下锁认证一下就好。常见的方法有版本号控制和时间戳控制。

悲观锁

悲观的认为数据会被其他事务修改,所以在拿到数据时直接加上锁,阻止其他事务进行修改。例如for update。

【MySQL】事务隔离级别及ACID的更多相关文章

  1. Mysql事务-隔离级别

    MYSQL事务-隔离级别 事务是什么? 事务简言之就是一组SQL执行要么全部成功,要么全部失败.MYSQL的事务在存储引擎层实现. 事务都有ACID特性: 原子性(Atomicity):一个事务必须被 ...

  2. MySQL事务隔离级别(一)

    本文实验的测试环境:Windows 10+cmd+MySQL5.6.36+InnoDB 一.事务的基本要素(ACID) 1.原子性(Atomicity):事务开始后所有操作,要么全部做完,要么全部不做 ...

  3. JDBC之事务隔离级别以及ACID特性

    JDBC之事务隔离级别以及ACID特性 事务隔离级别: 1.更新遗失(Lost update) 两个事务都同时更新一行数据,但是第二个事务却中途失败退出,导致对数据的两个修改都失效了.这是因为系统没有 ...

  4. [51CTO]新说MySQL事务隔离级别!

    新说MySQL事务隔离级别! 事务隔离级别这个问题,无论是校招还是社招,面试官都爱问!然而目前网上很多文章,说句实在话啊,我看了后我都怀疑作者弄懂没!本文所讲大部分内容,皆有官网作为佐证,因此对本文内 ...

  5. 查询mysql事务隔离级别

    查询mysql事务隔离级别 查询mysql事务隔离级别 分类: DB2011-11-26 13:12 2517人阅读 评论(0) 收藏 举报 mysqlsessionjava   1.查看当前会话隔离 ...

  6. MySQL事务隔离级别测试实例

    https://www.cnblogs.com/huanongying/p/7021555.html MySQL事务隔离级别 事务隔离级别 脏读 不可重复读 幻读 读未提交(read-uncommit ...

  7. MySQL事务隔离级别 解决并发问题

    MySQL事务隔离级别 1. 脏读: 骗钱的手段, 两个窗口或线程分别调用数据库转账表,转账后未提交,对方查看到账后,rollback,实际钱没转. 演示方法: mysql默认的事务隔离级别为repe ...

  8. mysql事务隔离级别、脏读、幻读

    Mysql事务隔离级别本身很重要,再加上可能是因为各大公司面试必问的缘故,在博客中出现的概率非常高,但不幸的是,中国的技术博客要么是转载,要么是照抄,质量参差不齐,好多结论都是错的,对于心怀好奇之心想 ...

  9. mysql事务隔离级别与设置

    mysql数据库,当且仅当引擎是InnoDB,才支持事务: 1.隔离级别 事务的隔离级别分为:未提交读(read uncommitted).已提交读(read committed).可重复读(repe ...

  10. MySQL事务隔离级别(二)

    搞清楚MySQL事务隔离级别 首先创建一个表 account.创建表的过程略过(由于 InnoDB 存储引擎支持事务,所以将表的存储引擎设置为 InnoDB).表的结构如下: 为了说明问题,我们打开两 ...

随机推荐

  1. 松软科技课堂:SQL--UNIQUE约束

    SQL UNIQUE 约束(文章来源:松软科技-www.sysoft.net.cn-) UNIQUE 约束唯一标识数据库表中的每条记录. UNIQUE 和 PRIMARY KEY 约束均为列或列集合提 ...

  2. 〈三〉ElasticSearch的认识:搜索、过滤、排序

    目录 上节回顾 本节前言 文档的搜索 URL参数条件搜索 请求体条件搜索 语法与示例: 补充: 小节总结: 文档的过滤filter 语法与举例: filter与bool constant_score ...

  3. Java第二次作业第三题

    四叶玫瑰线的图形设计:当用鼠标拖拽改变窗口大小时,四叶玫瑰线会重新绘制 package naizi; import java.awt.*; import java.awt.event.*; impor ...

  4. 初识数据库(MySql)

    一.简介 1.MySql是关系型数据库. 2.是一种开放源码软件, 3.是一种关联数据库管理系统. 4.服务器工作于客户端/服务端模式之下,或者是嵌入系统中. 数据库管理软件分类: 分两大类: 关系型 ...

  5. Day 25 网络基础2

    ip地址划分 ip地址由32位二进制组成. 什么是二进制?0101 10进制 0-9 10 2进制 0-1 10 二进制1100 ==1X2^3+1X2^2+0X2^1+0X2^0 =12 十进制15 ...

  6. 如何基于String实现锁?

    在某些时候,我们可能想基于字符串做一些事情,比如:针对同一用户的并发同步操作,使用锁字符串的方式实现比较合理.因为只有在相同字符串的情况下,并发操作才是不被允许的. 因为String 类型的变量赋值是 ...

  7. Promise.all结合数组Map用法

    Promise.all(iterable) 方法返回一个 Promise 实例,此实例在 iterable 参数内所有的 promise 都“完成(resolved)”或参数中不包含 promise  ...

  8. 阿里云搭建nginx + uWSGI 实现 django 项目

    系统版本 CentOS/7 64位 1.安装使用python3 创建python3目录 sudo mkdir /usr/local/python3 进入python3目录 cd /usr/local/ ...

  9. 【ADO.NET基础】后台获取前台控件

    C# 后台获取前台 input 文本框值.string aa=Request.Form[headself]; 那么要是后台给前台input文本框赋值呢? 后台 public string Headse ...

  10. JQuery 数组按指定长度分组

    JQuery方法 // 将data每3个一组进行分组 var data = ['法国','澳大利亚','智利','新西兰','西班牙','加拿大','阿根廷','美国','0','国产','波多黎各' ...