## 测试环境
mysql> select version();
+------------+
| version() |
+------------+
| 5.7.11-log |
+------------+

数据库事务特性 ACID,即

A(Atomicity)   -原子性

C(Consistency)- 一致性

I(Isolation)     - 隔离性

D(Durability)   - 持久性

MySQL 提供了 4 种不同的隔离级别,用来支持多版本并发控制(MVCC,Multi-Version Concurrency Control)。

默认的事务隔离级别是 REPEATABLE-READ(可重读):

mysql> select @@global.tx_isolation, @@session.tx_isolation;
+-------------------------+---------------------------+
| @@global.tx_isolation | @@session.tx_isolation |
+-------------------------+---------------------------+
| REPEATABLE-READ | REPEATABLE-READ |
+-------------------------+---------------------------+

在该事务级别下,一个事务期间内,该事务不考虑其他提交语句。

0x00、测试准备

1. 创建测试表

mysql> CREATE TABLE `transaction_test` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`val` varchar(20) NOT NULL,
`created` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1

2. 开启两个 MySQL 客户端进行测试

0x01、REPEATABLE-READ(可重读)

step 1:

在 Client 1 下开启事务,查询测试表中的数据:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> select * from transaction_test;
Empty set (0.00 sec)

step 2:

在 Client 2 下开启事务,并且往测试表中插入数据,但不提交事务:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> insert into transaction_test (val) values ('x'),('y'),('z');
Query OK, 3 rows affected (0.00 sec)
Records: 3 Duplicates: 0 Warnings: 0 mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 1 | x | 2017-02-06 00:20:59 |
| 2 | y | 2017-02-06 00:20:59 |
| 3 | z | 2017-02-06 00:20:59 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

step 3:

在 Client 1 下查看表中数据:

mysql> select * from transaction_test;
Empty set (0.00 sec)

仍然是空表。

step 4:

Client 2 提交事务:

mysql> commit;
Query OK, 0 rows affected (0.12 sec)

step 5:

Client 1 下查看表中数据:

mysql> select * from transaction_test;
Empty set (0.00 sec)

任然是空表。

step 6:

Client 1 提交事务,查看表中数据:

mysql> commit;
Query OK, 0 rows affected (0.00 sec) mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 1 | x | 2017-02-06 00:20:59 |
| 2 | y | 2017-02-06 00:20:59 |
| 3 | z | 2017-02-06 00:20:59 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

当 Client 1 完成事务后,才能看到其他事务提交的数据。

0x02、READ-COMMITTED(读取提交内容)

step 1:

Client 1 中清空表,改变数据库隔离级别:

mysql> truncate table transaction_test;
Query OK, 0 rows affected (0.10 sec) mysql> set @@session.tx_isolation = 'READ-COMMITTED';
Query OK, 0 rows affected (0.00 sec) mysql> select @@global.tx_isolation, @@session.tx_isolation;
+-----------------------+------------------------+
| @@global.tx_isolation | @@session.tx_isolation |
+-----------------------+------------------------+
| REPEATABLE-READ | READ-COMMITTED |
+-----------------------+------------------------+
1 row in set (0.00 sec)

step 2:

Client 1 开启事务,查询表中数据:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> select * from transaction_test;
Empty set (0.00 sec)

step 3:

Client 2 开启事务,向表中插入数据,但不提交事务:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> insert into transaction_test (val) values ('x'),('y'),('z');
Query OK, 3 rows affected (0.00 sec)
Records: 3 Duplicates: 0 Warnings: 0 mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 1 | x | 2017-02-06 00:31:00 |
| 2 | y | 2017-02-06 00:31:00 |
| 3 | z | 2017-02-06 00:31:00 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

step 4:

Client 1 下查看表中数据:

mysql> select * from transaction_test;
Empty set (0.00 sec)

仍然是空表。

step 5:

Client 2 提交事务:

mysql> commit;
Query OK, 0 rows affected (0.13 sec)

step 6:

Client 1 下查看表中数据:

mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 1 | x | 2017-02-06 00:31:00 |
| 2 | y | 2017-02-06 00:31:00 |
| 3 | z | 2017-02-06 00:31:00 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

与 REPEATABLE-READ 不同的是,Client 1 没有结束事务也能看到其他事务提交的数据。

0x03、READ-UNCOMMITTED(读取未提交内容)

step 1:

Client 1 下清空表,设置隔离级别:

mysql> truncate table transaction_test;
Query OK, 0 rows affected (0.10 sec) mysql> set @@session.tx_isolation = 'READ-UNCOMMITTED';
Query OK, 0 rows affected (0.00 sec) mysql> select @@global.tx_isolation, @@session.tx_isolation;
+-----------------------+------------------------+
| @@global.tx_isolation | @@session.tx_isolation |
+-----------------------+------------------------+
| REPEATABLE-READ | READ-UNCOMMITTED |
+-----------------------+------------------------+
1 row in set (0.00 sec)

step 2:

Client 1 下开启事务,查询表数据:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> select * from transaction_test;
Empty set (0.00 sec)

step 3:

Client 2 下开启事务,向表中插入数据,但是不提交事务:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> insert into transaction_test (val) values ('x'),('y'),('z');
Query OK, 3 rows affected (0.00 sec)
Records: 3 Duplicates: 0 Warnings: 0 mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 1 | x | 2017-02-06 00:43:59 |
| 2 | y | 2017-02-06 00:43:59 |
| 3 | z | 2017-02-06 00:43:59 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

step 4:

Client 1 中查询数据:

mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 1 | x | 2017-02-06 00:43:59 |
| 2 | y | 2017-02-06 00:43:59 |
| 3 | z | 2017-02-06 00:43:59 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

与 READ-COMMITTED 不同的是,在 Client 2 不提交事务的情况下,Client 1 也能读到其他事务插入的数据,即脏数据或者说产生了“脏读”。在一个事务期间读到了另一个事务在未提交之前产生的数据,那么第一个事务就读到了脏数据,产生了对第二个事务未提交数据的依赖,如果第二个事务回滚,那么第一个事务读到的数据是错误的脏数据。

“脏读”与“幻读”、“不可重复读”的区别是:幻读是读取结果集条数的对比,一个事务按相同的查询条件查询之前检索过的数据,发现检索出来的结果集条数变多或者减少(由其他事务插入、删除的),类似产生幻觉。

不可重复读是读取的数据本身的对比,一个事务在读取某些数据后的一段时间后,再次读取这个数据,发现其读取出来的数据内容已经发生了改变,就是不可重复读。

step 5:

Client 2 回滚事务:

mysql> rollback;
Query OK, 0 rows affected (0.04 sec) mysql> select * from transaction_test;
Empty set (0.00 sec)

step 6:

Client 1 查询表数据:

mysql> select * from transaction_test;
Empty set (0.00 sec)

空表。

0x04、SERIALIZABLE(序列化)

step 1:

Client 1 下清空表,设置隔离级别:

mysql> truncate table transaction_test;
Query OK, 0 rows affected (0.21 sec) mysql> set @@session.tx_isolation ='SERIALIZABLE';
Query OK, 0 rows affected (0.00 sec) mysql> select @@global.tx_isolation, @@session.tx_isolation;
+-----------------------+------------------------+
| @@global.tx_isolation | @@session.tx_isolation |
+-----------------------+------------------------+
| REPEATABLE-READ | SERIALIZABLE |
+-----------------------+------------------------+
1 row in set (0.00 sec)

step 2:

Client 1 开启事务,查询表:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> select * from transaction_test;
Empty set (0.00 sec)

step 3:

Client 2 开启事务,向表中插入数据:

mysql> start transaction;
Query OK, 0 rows affected (0.00 sec) mysql> insert into transaction_test (val) values ('x'),('y'),('z');
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction

此时 Client 2 插入数据(INSERT 操作)会被阻塞,直到第一个(Client 1)事务提交后,Client 2 的插入操作才能完成。

step 4:

Client 1 提交事务:

mysql> commit;
Query OK, 0 rows affected (0.00 sec)

step 5:

Client 2 插入数据:

mysql> insert into transaction_test (val) values ('x'),('y'),('z');
Query OK, 3 rows affected (0.00 sec)
Records: 3 Duplicates: 0 Warnings: 0 mysql> select * from transaction_test;
+----+-----+---------------------+
| id | val | created |
+----+-----+---------------------+
| 4 | x | 2017-02-06 00:54:17 |
| 5 | y | 2017-02-06 00:54:17 |
| 6 | z | 2017-02-06 00:54:17 |
+----+-----+---------------------+
3 rows in set (0.00 sec)

参考:

[MySQL]对于事务并发处理带来的问题,脏读、不可重复读、幻读的理解

MySQL 4 种隔离级别的区别的更多相关文章

  1. MySQL两种引擎的区别

    Innodb引擎 Innodb引擎提供了对数据库ACID事务的支持,并且实现了SQL标准的四种隔离级别.该引擎还提供了行级锁和外键约束,它的设计目标是处理大容量数据库系统,它本身其实就是基于MySQL ...

  2. Mysql 两种引擎的区别

    MyISAM与InnoDB的区别是什么? 1. 存储结构 MyISAM:每个MyISAM在磁盘上存储成三个文件.第一个文件的名字以表的名字开始,扩展名指出文件类型..frm文件存储表定义.数据文件的扩 ...

  3. mysql几种连接方式区别

    mysql的几种join 2017年03月19日 14:49:07 carl-zhao 阅读数:7845 标签: mysqlsqljoin 更多 个人分类: MySQL 版权声明:本文为博主原创文章, ...

  4. mysql几种关联的区别

    1.平时都是用的逗号的模式:select * from a,b where a.id=b.id,逗号的模式等于inner join和join: 2.left join 和 right join相反,效 ...

  5. MySQL ACID及四种隔离级别的解释

    以下内容出自<高性能MySQL>第三版,了解事务的ACID及四种隔离级有助于我们更好的理解事务运作. 下面举一个银行应用是解释事务必要性的一个经典例子.假如一个银行的数据库有两张表:支票表 ...

  6. [转]Mysql几种索引类型的区别及适用情况

    此为转载文章,仅做记录使用,方便日后查看,原文链接:https://www.cnblogs.com/yuan-shuai/p/3225417.html Mysql几种索引类型的区别及适用情况   如大 ...

  7. MySQL中四种隔离级别的演示

    事务的隔离是并发操作中需要理解清楚的问题.MySQL中共有4种不同的隔离级别,这4种隔离级别分别是: 隔离级别类型 影响结果 READ UNCOMMITTED(未提交读) 事务将会读取到未提交的数据, ...

  8. MySQL数据库引擎介绍、区别、创建和性能测试的深入分析

    本篇文章是对MySQL数据库引擎介绍.区别.创建和性能测试进行了详细的分析介绍,需要的朋友参考下   数据库引擎介绍 MySQL数据库引擎取决于MySQL在安装的时候是如何被编译的.要添加一个新的引擎 ...

  9. [转]MySQL数据库引擎介绍、区别、创建和性能测试的深入分析

    本篇文章是对MySQL数据库引擎介绍.区别.创建和性能测试进行了详细的分析介绍,需要的朋友参考下   数据库引擎介绍 MySQL数据库引擎取决于MySQL在安装的时候是如何被编译的.要添加一个新的引擎 ...

随机推荐

  1. gdb windbg and od use

    gdb aslr -- 显示/设置 gdb 的 ASLR asmsearch -- Search for ASM instructions in memory asmsearch "int ...

  2. SpringMVC(十一) RequestMapping获取Cookie值

    可以在控制器方法中使用类似@CookieValue("JSESSIONID") String sessionID的方式,来获取请求中的Cookie的值. 样例控制器代码 packa ...

  3. Oracle DB

    索引 一.安装   卸载 一.安装 1.Oracle首先询问用户是否接受更新信息,一般选择不接受:Oracle安装选项,需为其配置数据库,所以此处会询问是否创建安装数据库,选择是系统类型,一般选择服务 ...

  4. 数据源、数据集、同步任务、数据仓库、元数据、数据目录、主题、来源系统、标签、增量识别字段、修改同步、ES索引、HBase列族、元数据同步、

    数据源.数据集.同步任务.数据仓库.元数据.数据目录.主题.来源系统.标签. 增量识别字段.修改同步.ES索引.HBase列族.元数据同步.DS.ODS.DW.DM.zk集群地址 == 数据源 数据源 ...

  5. 电脑用bat脚本给手机批量自动安装apk文件 autoInstall.bat

    ------创建这个文件autoInstall.bat 内容如下: @ECHO off @REM 将adb.exe添加到PATH中ECHO 初始化…@SET PATH=%PATH%;%CD%\Adb@ ...

  6. IE和DOM事件流

    * ie采用冒泡型事件 Netscape使用捕获型事件 dom使用先冒泡后捕获事件 冒泡型事件模型: button->div->body (IE事件流) 捕获型事件模型: body-> ...

  7. GDOI2017爆炸记

    100种方法教你爆零.. 总结 其实这一次比赛除了三个sb的错误还是收获到了很多的.. 起码自己已经知道自己有进队的实力 不足的地方很大 主要是脑子不太好使,题目要不只能拿最暴力的分要不就能a 看了很 ...

  8. shell脚本使用技巧3--调试

    1.使用-x,开启shell脚本的跟踪调试功能 ex:bash -x script.sh or sh -x script.sh 2.使用set -x 和 set +x对脚本进行部分调试(输入中间的内容 ...

  9. 函数的name属性

    匿名函数表达式的广泛使用加大了辨别函数的难度,所以ES6 中为所有函数新增了name属性 例如: name属性的特殊情况 (1) (2) 绑定函数的name属性总是由被绑定函数的name属性和字符串前 ...

  10. 区间dp板子题:[noi1995]石子合并

    非常经典的区间dp模板 对于每一个大于二的区间 我们显然都可以将它拆分成两个子序列 那么分别计算对于每个取最优值即可 #pragma GCC optimize("O2") #inc ...