在DBMS中,事務保證了一個操作序列可以全部都執行或者全部都不執行(原子性),從一個狀態轉變到另外一個狀態(一致性)。由於事務滿足久性。所以一旦事務被提交之後,數據就能夠被持久化下來,又因為事務是滿足隔離性的,所以,當多個事務同時處理同一個數據的時候,多個事務直接是互不影響的,所以,在多個事務並發操作的過程中,如果控制不好隔離級別,就有可能產生髒讀、不可重復讀或者幻讀等讀現象。
在數據庫事務的ACID四個屬性中,隔離性是一個最常放松的一個。可以在數據操作過程中利用數據庫的鎖機制或者多版本並發控制機制獲取更高的隔離等級。但是,隨著數據庫隔離級別的提高,數據的並發能力也會有所下降。所以,如何在並發性和隔離性之間做一個很好的權衡就成了一個至關重要的問題。
數據庫事務的隔離級別有4個,由低到高依次為Read uncommitted、Read committed、Repeatable read、Serializable,這四個級別可以逐個解決髒讀、不可重復讀、幻讀這幾類問題。
注意:我們討論隔離級別的場景,主要是在多個事務並發的情況下,因此,接下來的講解都圍繞事務並發。
讀未提交,顧名思義,就是一個事務可以讀取另一個未提交事務的數據。
事例:老板要給程序員發工資,程序員的工資是3.6萬/月。但是發工資時老板不小心按錯了數字,按成3.9萬/月,該錢已經打到程序員的戶口,但是事務還沒有提交,就在這時,程序員去查看自己這個月的工資,發現比往常多了3千元,以為漲工資了非常高興。但是老板及時發現了不對,馬上回滾差點就提交了的事務,將數字改成3.6萬再提交。
分析:這種情況就是髒讀,兩個並發的事務,"事務A":領導發工資、"事務B":程序員查看工資,事務B讀取了事務A尚未提交的數據實際程序員這個月的工資還是3.6萬,但是程序員看到的是3.9萬。
事務在讀數據的時候並未對數據加鎖。
事務在修改數據的時候只對數據增加行級共享鎖。
表現:
設置 set session transaction isolation level read uncommitted;
set autocommit=0;
下面還是借用我在數據庫的讀現象淺析一文中舉的例子來說明在未提交讀的隔離級別中兩個事務之間的隔離情況。
/* Query 1 */
SELECT age FROM users WHERE id = 1;
/* will read 20 */
/* Query 2 */
UPDATE users SET age = 21 WHERE id = 1;
/* No commit here */
/* Query 1 */
SELECT age FROM users WHERE id = 1;
/* will read 21 */
ROLLBACK;
/* lock-based DIRTY READ */
事務一共查詢了兩次,在兩次查詢的過程中,事務二對數據進行了修改,並未提交(commit)。但是事務一的第二次查詢查到了事務二的修改結果。在數據庫的讀現象淺析中我們介紹過,這種現象我們稱之為髒讀。
所以,未提交讀會導致髒讀
事務1更新某行記錄時,事務2不能對這行記錄做更新,直到事務1結束:
可以看到右邊更新一個記錄時,還沒有提交;
此時,左邊的事務二也對這條記錄更新,會阻塞。
提交讀,顧名思義,就是一個事務要等另一個事務提交後才能讀取數據。
事例:程序員拿著信用卡去享受生活(卡裡當然是只有3.6萬),當他埋單時(程序員事務開啟),收費系統事先檢測到他的卡裡有3.6萬,就在這個時候!!程序員的妻子要把錢全部轉出充當家用,並提交。當收費系統准備扣款時,再檢測卡裡的金額,發現已經沒錢了(第二次檢測金額當然要等待妻子轉出金額事務提交完)。程序員就會很郁悶,明明卡裡是有錢的…
分析:這就是提交讀,若有事務對數據進行更新(UPDATE)操作時,讀操作事務要等待這個更新操作事務提交後才能讀取數據,可以解決髒讀問題。但在這個事例中,出現了一個事務范圍內兩個相同的查詢卻返回了不同數據,這就是不可重復讀。
事務對當前被讀取的數據加 行級共享鎖(當讀到時才加鎖),一旦讀完該行,立即釋放該行級共享鎖;
事務在更新某數據的瞬間(就是發生更新的瞬間),必須先對其加 行級排他鎖,直到事務結束才釋放。
/* Query 1 */
SELECT * FROM users WHERE id = 1;
/* Query 2 */
UPDATE users SET age = 21 WHERE id = 1;
COMMIT;
/* in multiversion concurrency
control, or lock-based READ COMMITTED */
/* Query 1 */SELECT * FROM users WHERE id = 1
lock in share mode;
COMMIT; /*lock-based REPEATABLE READ */
**在事務二沒有提交之前,事務一是不能更改這條記錄的,也不可以加共享鎖(lock in share mode)。只能執行普通的查詢。
在提交讀隔離級別中,在事務二提交之前,事務一不能讀取數據。只有在事務二提交之後,事務一才能讀數據。
但是從上面的例子中我們也看到,事務一兩次讀取的結果並不一致,所以提交讀不能解決不可重復讀的讀現象。
簡而言之,提交讀這種隔離級別保證了讀到的任何數據都是提交的數據,避免了髒讀(dirty reads)。但是不保證事務重新讀的時候能讀到相同的數據,因為在每次數據讀完之後其他事務可以修改剛才讀到的數據。
重復讀,就是在開始讀取數據(事務開啟)時,不再允許修改操作
事例:程序員拿著信用卡去享受生活(卡裡當然是只有3.6萬),當他買單時(事務開啟,不允許其他事務的UPDATE修改操作),收費系統事先檢測到他的卡裡有3.6萬。這個時候他的妻子不能轉出金額了。接下來收費系統就可以扣款了。
分析:重復讀可以解決不可重復讀問題。寫到這裡,應該明白的一點就是,不可重復讀對應的是修改,即UPDATE操作。但是可能還會有幻讀問題。因為幻讀問題對應的是插入INSERT操作,而不是UPDATE操作。
事務在讀取某數據的瞬間(就是開始讀取的瞬間),必須先對其加 行級共享鎖,直到事務結束才釋放;
事務在更新某數據的瞬間(就是發生更新的瞬間),必須先對其加 行級排他鎖,直到事務結束才釋放。
現象
事務1在讀取某行記錄的整個過程中,事務2都可以對該行記錄進行讀取(因為事務一對該行記錄增加行級共享鎖的情況下,事務二同樣可以對該數據增加共享鎖來讀數據。)。
事務1在讀取某行記錄的整個過程中,事務2都不能修改該行數據(事務一在讀取的整個過程會對數據增加共享鎖,直到事務提交才會釋放鎖,所以整個過程中,任何其他事務都不能對該行數據增加排他鎖。所以,可重復讀能夠解決不可重復讀
的讀現象)
/* Query 1 */
SELECT * FROM users WHERE id = 1;
COMMIT;
/* Query 2 */
UPDATE users SET age = 21 WHERE id = 1;
COMMIT;
/* in multiversion concurrency
control, or lock-based READ COMMITTED */
在上面的例子中,只有在事務一提交之後,事務二才能更改該行數據。所以,只要在事務一從開始到結束的這段時間內,無論他讀取該行數據多少次,結果都是一樣的。
幻讀
事例:程序員某一天去消費,花了2千元,然後他的妻子去查看他今天的消費記錄(全表掃描FTS,妻子事務開啟),看到確實是花了2千元,就在這個時候,程序員花了1萬買了一部電腦,即新增INSERT了一條消費記錄,並提交。當妻子打印程序員的消費記錄清單時(妻子事務提交),發現花了1.2萬元,似乎出現了幻覺,這就是幻讀。
事務一:妻子去查看消費記錄,事務一的兩次范圍查詢結果並不相同。這也就是我們提到的幻讀。
可重復讀隔離級別可以解決不可重復讀的讀現象,但是解決不了幻讀。解決幻讀需要序列化。
Serializable 是最高的事務隔離級別,在該級別下,事務串行化順序執行,可以避免髒讀、不可重復讀與幻讀。但是這種事務隔離級別效率低下,比較耗數據庫性能,一般不使用。
事務在讀取數據時,必須先對其加 表級共享鎖 ,直到事務結束才釋放;
事務在更新數據時,必須先對其加 表級排他鎖 ,直到事務結束才釋放。
值得一提的是:大多數數據庫默認的事務隔離級別是Read committed,比如Sql Server , Oracle。MySQL的默認隔離級別是Repeatable read。
參考文章:
深入分析事務的隔離級別
理解事務的4種隔離級別
轉發請注明出處:http://www.cnblogs.com/jycboy/p/transaction.html