自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

數(shù)據(jù)庫(kù) MySQL
事務(wù)是邏輯上的一組操作,要么都執(zhí)行,要么都不執(zhí)行。

 什么是事務(wù)?

事務(wù)是邏輯上的一組操作,要么都執(zhí)行,要么都不執(zhí)行。

[[315803]]

事務(wù)最經(jīng)典也經(jīng)常被拿出來(lái)說(shuō)例子就是轉(zhuǎn)賬了。假如小明要給小紅轉(zhuǎn)賬1000元,這個(gè)轉(zhuǎn)賬會(huì)涉及到兩個(gè)關(guān)鍵操作就是:將小明的余額減少1000元,將小紅的余額增加1000元。萬(wàn)一在這兩個(gè)操作之間突然出現(xiàn)錯(cuò)誤比如銀行系統(tǒng)崩潰,導(dǎo)致小明余額減少而小紅的余額沒(méi)有增加,這樣就不對(duì)了。事務(wù)就是保證這兩個(gè)關(guān)鍵操作要么都成功,要么都要失敗。

事物的特性(ACID)

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

原子性: 事務(wù)是最小的執(zhí)行單位,不允許分割。事務(wù)的原子性確保動(dòng)作要么全部完成,要么完全不起作用;

一致性: 執(zhí)行事務(wù)前后,數(shù)據(jù)保持一致;

隔離性: 并發(fā)訪(fǎng)問(wèn)數(shù)據(jù)庫(kù)時(shí),一個(gè)用戶(hù)的事物不被其他事物所干擾,各并發(fā)事務(wù)之間數(shù)據(jù)庫(kù)是獨(dú)立的;

持久性: 一個(gè)事務(wù)被提交之后。它對(duì)數(shù)據(jù)庫(kù)中數(shù)據(jù)的改變是持久的,即使數(shù)據(jù)庫(kù)發(fā)生故障也不應(yīng)該對(duì)其有任何影響。

并發(fā)事務(wù)帶來(lái)的問(wèn)題

在典型的應(yīng)用程序中,多個(gè)事務(wù)并發(fā)運(yùn)行,經(jīng)常會(huì)操作相同的數(shù)據(jù)來(lái)完成各自的任務(wù)(多個(gè)用戶(hù)對(duì)統(tǒng)一數(shù)據(jù)進(jìn)行操作)。并發(fā)雖然是必須的,但可能會(huì)導(dǎo)致以下的問(wèn)題。

臟讀(Dirty read): 當(dāng)一個(gè)事務(wù)正在訪(fǎng)問(wèn)數(shù)據(jù)并且對(duì)數(shù)據(jù)進(jìn)行了修改,而這種修改還沒(méi)有提交到數(shù)據(jù)庫(kù)中,這時(shí)另外一個(gè)事務(wù)也訪(fǎng)問(wèn)了這個(gè)數(shù)據(jù),然后使用了這個(gè)數(shù)據(jù)。因?yàn)檫@個(gè)數(shù)據(jù)是還沒(méi)有提交的數(shù)據(jù),那么另外一個(gè)事務(wù)讀到的這個(gè)數(shù)據(jù)是“臟數(shù)據(jù)”,依據(jù)“臟數(shù)據(jù)”所做的操作可能是不正確的。

丟失修改(Lost to modify): 指在一個(gè)事務(wù)讀取一個(gè)數(shù)據(jù)時(shí),另外一個(gè)事務(wù)也訪(fǎng)問(wèn)了該數(shù)據(jù),那么在第一個(gè)事務(wù)中修改了這個(gè)數(shù)據(jù)后,第二個(gè)事務(wù)也修改了這個(gè)數(shù)據(jù)。這樣第一個(gè)事務(wù)內(nèi)的修改結(jié)果就被丟失,因此稱(chēng)為丟失修改。例如:事務(wù)1讀取某表中的數(shù)據(jù)A=20,事務(wù)2也讀取A=20,事務(wù)1修改A=A-1,事務(wù)2也修改A=A-1,最終結(jié)果A=19,事務(wù)1的修改被丟失。

不可重復(fù)讀(Unrepeatableread): 指在一個(gè)事務(wù)內(nèi)多次讀同一數(shù)據(jù)。在這個(gè)事務(wù)還沒(méi)有結(jié)束時(shí),另一個(gè)事務(wù)也訪(fǎng)問(wèn)該數(shù)據(jù)。那么,在第一個(gè)事務(wù)中的兩次讀數(shù)據(jù)之間,由于第二個(gè)事務(wù)的修改導(dǎo)致第一個(gè)事務(wù)兩次讀取的數(shù)據(jù)可能不太一樣。這就發(fā)生了在一個(gè)事務(wù)內(nèi)兩次讀到的數(shù)據(jù)是不一樣的情況,因此稱(chēng)為不可重復(fù)讀。

幻讀(Phantom read): 幻讀與不可重復(fù)讀類(lèi)似。它發(fā)生在一個(gè)事務(wù)(T1)讀取了幾行數(shù)據(jù),接著另一個(gè)并發(fā)事務(wù)(T2)插入了一些數(shù)據(jù)時(shí)。在隨后的查詢(xún)中,第一個(gè)事務(wù)(T1)就會(huì)發(fā)現(xiàn)多了一些原本不存在的記錄,就好像發(fā)生了幻覺(jué)一樣,所以稱(chēng)為幻讀。

不可重復(fù)度和幻讀區(qū)別:

不可重復(fù)讀的重點(diǎn)是修改,幻讀的重點(diǎn)在于新增或者刪除。

例1(同樣的條件, 你讀取過(guò)的數(shù)據(jù), 再次讀取出來(lái)發(fā)現(xiàn)值不一樣了 ):事務(wù)1中的A先生讀取自己的工資為 1000的操作還沒(méi)完成,事務(wù)2中的B先生就修改了A的工資為2000,導(dǎo) 致A再讀自己的工資時(shí)工資變?yōu)?2000;這就是不可重復(fù)讀。

例2(同樣的條件, 第1次和第2次讀出來(lái)的記錄數(shù)不一樣 ):假某工資單表中工資大于3000的有4人,事務(wù)1讀取了所有工資大于3000的人,共查到4條記錄,這時(shí)事務(wù)2 又插入了一條工資大于3000的記錄,事務(wù)1再次讀取時(shí)查到的記錄就變?yōu)榱?條,這樣就導(dǎo)致了幻讀。

事務(wù)隔離級(jí)別

SQL 標(biāo)準(zhǔn)定義了四個(gè)隔離級(jí)別:

READ-UNCOMMITTED(讀取未提交): 最低的隔離級(jí)別,允許讀取尚未提交的數(shù)據(jù)變更,可能會(huì)導(dǎo)致臟讀、幻讀或不可重復(fù)讀

READ-COMMITTED(讀取已提交): 允許讀取并發(fā)事務(wù)已經(jīng)提交的數(shù)據(jù),可以阻止臟讀,但是幻讀或不可重復(fù)讀仍有可能發(fā)生

REPEATABLE-READ(可重讀): 對(duì)同一字段的多次讀取結(jié)果都是一致的,除非數(shù)據(jù)是被本身事務(wù)自己所修改,可以阻止臟讀和不可重復(fù)讀,但幻讀仍有可能發(fā)生。

SERIALIZABLE(可串行化): 最高的隔離級(jí)別,完全服從ACID的隔離級(jí)別。所有的事務(wù)依次逐個(gè)執(zhí)行,這樣事務(wù)之間就完全不可能產(chǎn)生干擾,也就是說(shuō),該級(jí)別可以防止臟讀、不可重復(fù)讀以及幻讀

MySQL InnoDB 存儲(chǔ)引擎的默認(rèn)支持的隔離級(jí)別是 REPEATABLE-READ(可重讀)。我們可以通過(guò)SELECT @@tx_isolation;命令來(lái)查看

mysql> SELECT @@tx_isolation;
+-----------------+
| @@tx_isolation |
+-----------------+
| REPEATABLE-READ |
+-----------------+

這里需要注意的是:與 SQL 標(biāo)準(zhǔn)不同的地方在于InnoDB 存儲(chǔ)引擎在 REPEATABLE-READ(可重讀)事務(wù)隔離級(jí)別下使用的是Next-Key Lock 鎖算法,因此可以避免幻讀的產(chǎn)生,這與其他數(shù)據(jù)庫(kù)系統(tǒng)(如 SQL Server)是不同的。所以說(shuō)InnoDB 存儲(chǔ)引擎的默認(rèn)支持的隔離級(jí)別是 REPEATABLE-READ(可重讀) 已經(jīng)可以完全保證事務(wù)的隔離性要求,即達(dá)到了 SQL標(biāo)準(zhǔn)的SERIALIZABLE(可串行化)隔離級(jí)別。

因?yàn)楦綦x級(jí)別越低,事務(wù)請(qǐng)求的鎖越少,所以大部分?jǐn)?shù)據(jù)庫(kù)系統(tǒng)的隔離級(jí)別都是READ-COMMITTED(讀取提交內(nèi)容):,但是你要知道的是InnoDB 存儲(chǔ)引擎默認(rèn)使用REPEATABLE-READ(可重讀)并不會(huì)有任何性能損失。

InnoDB 存儲(chǔ)引擎在 分布式事務(wù) 的情況下一般會(huì)用到SERIALIZABLE(可串行化)隔離級(jí)別。

實(shí)際情況演示

MySQL 命令行的默認(rèn)配置中事務(wù)都是自動(dòng)提交的,即執(zhí)行SQL語(yǔ)句后就會(huì)馬上執(zhí)行 COMMIT 操作。如果要顯式地開(kāi)啟一個(gè)事務(wù)需要使用命令:START TARNSACTION。

我們可以通過(guò)下面的命令來(lái)設(shè)置隔離級(jí)別。

SET [SESSION|GLOBAL] TRANSACTION ISOLATION LEVEL [READ UNCOMMITTED|READ COMMITTED|REPEATABLE READ|SERIALIZABLE]

我們?cè)賮?lái)看一下我們?cè)谙旅鎸?shí)際操作中使用到的一些并發(fā)控制語(yǔ)句:

•START TARNSACTION |BEGIN:顯式地開(kāi)啟一個(gè)事務(wù)。

•COMMIT:提交事務(wù),使得對(duì)數(shù)據(jù)庫(kù)做的所有修改成為永久性。

•ROLLBACK 回滾會(huì)結(jié)束用戶(hù)的事務(wù),并撤銷(xiāo)正在進(jìn)行的所有未提交的修改。

在下面我會(huì)使用 2 個(gè)命令行 MySQL ,模擬多線(xiàn)程(多事務(wù))。

臟讀(讀未提交)

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

 

避免臟讀(讀已提交)

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

 

不可重復(fù)讀

還是剛才上面的讀已提交的圖,雖然避免了讀未提交,但是卻出現(xiàn)了,一個(gè)事務(wù)還沒(méi)有結(jié)束,就發(fā)生了 不可重復(fù)讀問(wèn)題。

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

 

可重復(fù)讀

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

 

防止幻讀(可重復(fù)讀)

一文帶你輕松搞懂事務(wù)隔離級(jí)別(圖文詳解)

 

一個(gè)事務(wù)對(duì)數(shù)據(jù)庫(kù)進(jìn)行操作,這種操作的范圍是數(shù)據(jù)庫(kù)的全部行,然后第二個(gè)事務(wù)也在對(duì)這個(gè)數(shù)據(jù)庫(kù)操作,這種操作可以是插入一行記錄或刪除一行記錄,那么第一個(gè)是事務(wù)就會(huì)覺(jué)得自己出現(xiàn)了幻覺(jué),怎么還有沒(méi)有處理的記錄呢? 或者 怎么多處理了一行記錄呢?

幻讀和不可重復(fù)讀有些相似之處 ,但是不可重復(fù)讀的重點(diǎn)是修改,幻讀的重點(diǎn)在于新增或者刪除。

責(zé)任編輯:武曉燕 來(lái)源: 今日頭條
相關(guān)推薦

2021-10-19 10:10:51

MySQL事務(wù)隔離級(jí)別數(shù)據(jù)庫(kù)

2023-03-06 21:29:41

mmap技術(shù)操作系統(tǒng)

2010-11-19 16:13:06

oracle事務(wù)隔離級(jí)

2020-04-07 09:21:45

MySQL數(shù)據(jù)庫(kù)SQL

2021-09-07 09:46:40

JavaScriptGenerator函數(shù)

2021-09-11 10:41:27

PythonPickle模塊

2024-11-19 13:20:55

2022-05-11 11:54:55

Http傳送協(xié)議

2021-08-05 06:54:05

觀察者訂閱設(shè)計(jì)

2021-12-01 11:40:14

Python 輸入輸出

2023-03-31 13:01:31

PythonCelery驗(yàn)證

2021-11-06 10:18:30

Python變量常量

2024-12-02 08:37:04

2021-10-13 21:43:18

JVMRPC框架

2024-04-12 12:19:08

語(yǔ)言模型AI

2022-03-24 08:51:48

Redis互聯(lián)網(wǎng)NoSQL

2021-09-28 07:12:10

avaScriptCurrying柯里化

2021-07-21 09:24:25

MongoDB數(shù)據(jù)庫(kù) Python

2020-03-18 13:40:03

Spring事數(shù)據(jù)庫(kù)代碼

2019-04-03 09:27:01

MySQLInnoDB務(wù)ACID
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)