# MySQL-事务介绍

---

[TOC]

---

# MySQL事务

## ACID

1.**原子性(Atomcity)**

一个事务的最小单元,要么全部成功要么全部失败,执行的过程中是不能被打断或者执行其他操作的。

2.**一致性(Consistent)**

事务开始前和结束后,数据库的完整性约束没有被破坏。比如A向B转账,不可能A扣了钱,B却没收到,事务开始前A+B=500,事务结束后A+B不可能!=500。

3.**隔离性(Isolation)**

隔离性表示各个事务之间不会互相影响,数据库一般会提供多种级别的隔离。实际上多个事务是并发执行的,但是他们之间不会互相影响。

4.**持久性(Durability)**

事务提交后,事务对数据库的所有更新将被保存到数据库,不能回滚。

##事务的隔离级别

事务分为以下4个级别

- **Read UnCommitted(可以读取未提交数据)**

- **Read Committed(只能读到已提交数据)**

- **Read Repeatable(一个事务中重复读取,数据保持一致性)**

- **Serializable(串行执行,不会造成不一致问题,但会影响并发)**

##不同的隔离级别可能引发不同的一致性问题

|隔离级别|脏读|不可重复读|幻读|

|:-:|:-:|:-:|:-:|

|Read UnCommitted|Y|Y|Y|

|Read Committed|N|Y|Y|

|Read Repeatable|N|N|Y|

|serializable|N|N|N|

##并发事务下可能导致的数据不一致

### 1. **脏读(Dirty Read)> 读取到中间值**

事务A开启事务,做更新操作数据由<font color="#FFA500">300</font>更新为<font color="#9ACD32">400</font>,并未commit,此时事务B会读取到A更新但未提交的值<font color="#9ACD32">400</font>。此时A事务 rollback,但是B事务已经读取到A更新的值,造成<font color="#CD0000">**脏读**</font>。

事例:

- **A连接**

```mysql

#设置当前会话事务隔离级别为读未提交

SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;

#开始执行事务

START TRANSACTION;

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

#②在事务B第一次查询之后进行更新操作

UPDATE tb_user_account SET balance = 400 WHERE user_id = 86;

#④发生异常进行回滚

ROLLBACK;

```

- **B连接**

```mysql

#查询当前连接事务级别

SELECT @@tx_isolation;

#设置当前会话事务隔离级别为读未提交

SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;

#开始执行事务

START TRANSACTION;

#①在事务A更新之前执行

SELECT * FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

#③在事务A更新之后执行,此时读取到了未提交的数据400

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    400 |

+---------+---------+

#⑤在事务A回滚之后执行,两次读取到的数据不一致,发生的了脏读

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

```

```flow

A=>operation: 事务A 更新为400

BS1=>operation: 事务B 第一次查询到300

BS2=>operation: 事务B 第二次查询到400(中间值)脏读

ROLL=>operation: 事务A 回滚

BS3=>operation: 事务B 第三次查询到300 (数据不一致)

BS1->A->BS2->ROLL->BS3

```

### 2. **不可重复读(UnRepeatable Read)> 更新场景,数据不一致**

事务A开启事务,做更新操作数据由<font color="#FFA500">300</font>更新为<font color="#9ACD32">400</font>,并未commit,此时事务B查询值为<font color="#FFA500">300</font>,解决了脏读问题。此时A提交事务,事务B再次查询值为<font color="#9ACD32">400</font>,两次查询数据不一致。(<font color="#CD0000">**不可重复读**</font>)

事例:

- **A连接**

```mysql

#设置当前会话事务隔离级别为读已提交

SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;

#开始执行事务

START TRANSACTION;

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

#②在事务B第一次查询之后进行更新操作

UPDATE tb_user_account SET balance = 400 WHERE user_id = 86;

#④提交数据

COMMIT;

```

- **B连接**

```mysql

#查询当前连接事务级别

SELECT @@tx_isolation;

#设置当前会话事务隔离级别为读未提交

SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;

#开始执行事务

START TRANSACTION;

#①在事务A提交之前执行

SELECT * FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

#③在事务A提交之后执行,此时读取到的值还是300,解决了脏读问题。

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

#⑤在事务A提交之后执行,两次读取到的数据不一致,发生的了不可重复读。

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    400 |

+---------+---------+

```

```flow

A=>operation: 事务A 更新为400

BS1=>operation: 事务B 第一次查询到300

BS2=>operation: 事务B 第二次查询到300,保证了不发生脏读

ROLL=>operation: 事务A 提交

BS3=>operation: 事务B 第三次查询到400 (不可重复读,数据不一致)

BS1->A->BS2->ROLL->BS3

```

### 3. **幻读(Phantom Read)>插入or删除场景,数据不一致**

事务A开启事务,做查询大于<font color="#FFA500">0</font>的数据,并未commit,此时B连接插入一条大于0的数据。A再次查询大于<font color="#FFA500">0</font>的数据,但是并查询到B插入的数据,造成(<font color="#CD0000">**幻读**</font>)。

事例:

- **A连接**

```mysql

#设置当前会话事务隔离级别为可重复读

SET SESSION TRANSACTION ISOLATION LEVEL REPEATABLE READ;

#开始执行事务

START TRANSACTION;

#①查询大于0的数据

SELECT count(1) FROM tb_user_account WHERE balance > 0;

+----------+

| count(1) |

+----------+

|        1 |

+----------+

#③查询大于0的数据,查到新增数据(造成幻读,BUT MySQL,MVVC解决了此问题)

SELECT count(1) FROM tb_user_account WHERE balance > 0;

+----------+

| count(1) |

+----------+

|        2 |

+----------+

```

- **B连接**

```mysql

#②在事务A提交之前执行

INSERT INTO tb_user_account(balance) VALUES(100);

```

```flow

A=>operation: 事务A  查询大于0的数据

BS1=>operation: 事务B 插入一条大于0的数据

AS2=>operation: 事务A  查询大于0的数据,查到了新增数据。(幻读,数据不一致)

A->BS1->ROLL->BS3

```

### 4. 更新丢失(lost update)>幻读中数据更新丢失

使用 READ COMMITTED隔离级别,事务A开启事务,做更新操作数据增加<font color="#FFA500">100</font>,并未commit,此时事务B也做更新操作数据增加<font color="#9ACD32">100</font>。此时B先提交事务,之后A再提交事务,会造成B事务**<font color="#CD0000">更新丢失</font>**。(MySQL的InnoDB使用了MVCC,在提交B时会锁住行数据,避免此种问题)。

事例:

- **A连接**

```mysql

#设置当前会话事务隔离级别为读已提交

SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;

#开始执行事务

START TRANSACTION;

SELECT user_id,balance FROM tb_user_account;

+---------+---------+

| user_id | balance |

+---------+---------+

|      80 |      0 |

|      86 |    300 |

+---------+---------+

#①更新操作数据增加100。

UPDATE tb_user_account SET balance = balance + 100 WHERE user_id = 86;

#④提交数据,此时会造成事务B的更新丢失。

COMMIT;

```

- **B连接**

```mysql

#查询当前连接事务级别

SELECT @@tx_isolation;

#设置当前会话事务隔离级别为读未提交

SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;

#开始执行事务

START TRANSACTION;

#②在事务A提交之前执行,增加100。

UPDATE tb_user_account SET balance = balance + 100 WHERE user_id = 86;

#③事务B提交更新

COMMIT;

```

```flow

A=>operation: 事务A 增加100

BS1=>operation: 事务B 增加100

BCOMMIT=>operation: 事务B 提交

ACOMMIT=>operation: 事务A 提交(A未读取到B的更新,造成更新丢失)

A->BS1->BCOMMIT->ACOMMIT

```

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容