Mysql InnoDB的四個事務隔離級別和(分別逐級解決的問題)臟讀,不可重復讀,虛讀


MySqlInnoDB的事務隔離級別有四個:(默認是可重復讀repeatable read)

未提交讀 read uncommit : 在另一個事務修改了數據,但尚未提交,在本事務中SELECT語句可能會查詢到這些未被提交的數據,而發生臟讀。

提交讀 read commit : 在一個事務中發生兩次SELECT查詢,當第一次SELECT執行完查詢到一些數據,接下來另一個事務修改了這些數據並提交了,當第二次SELECT執行的時候查詢到的數據和第一次SELECT的不同,而發生不可重復讀、幻讀問題,但解決了臟讀(鎖定所讀取的當前行)。

可重復讀 repeatable read:在同一個事務中,SELECT的結果是事務開啟時時間點的結果,因此,同樣的SELECT查詢的結果總是一致的。解決了不可重復讀,但有可能發生虛讀(鎖定所讀取的所有行)。

串行化 serializable:

(轉載的)

    1). 臟讀 

      首先區分臟頁和臟數據

      臟頁是內存的緩沖池中已經修改的page,未及時flush到硬盤,但已經寫到redo log中。讀取和修改緩沖池的page很正常,可以提高效率,flush即可同步。臟數據是指事務對緩沖池中的行記錄record進行了修改,但是還沒提交!!!,如果這時讀取緩沖池中未提交的行數據就叫臟讀,違反了事務的隔離性。臟讀就是指當一個事務正在訪問數據,並且對數據進行了修改,而這種修改還沒有提交到數據庫中,這時,另外一個事務也訪問這個數據,然后使用了這個數據。

    2). 不可重復讀 

     是指在一個事務內,多次讀同一數據。在這個事務還沒有結束時,另外一個事務也訪問該同一數據。那么,在第一個事務中的兩次讀數據之間,由於第二個事務的修改,第二個事務已經提交。那么第一個事務兩次讀到的的數據可能是不一樣的。這樣就發生了在一個事務內兩次讀到的數據是不一樣的,因此稱為是不可重復讀。例如,一個編輯人員兩次讀取同一文檔,但在兩次讀取之間,作者重寫了該文檔。當編輯人員第二次讀取文檔時,文檔已更改。原始讀取不可重復。如果只有在作者全部完成編寫后編輯人員才可以讀取文檔,則可以避免該問題

    3). 虛讀 :

     是指當事務不是獨立執行時發生的一種現象,例如第一個事務對一個表中的數據進行了修改,這種修改涉及到表中的全部數據行。同時,第二個事務也修改這個表中的數據,這種修改是向表中插入一行新數據。那么,以后就會發生操作第一個事務的用戶發現表中還有沒有修改的數據行,就好象發生了幻覺一樣。例如,一個編輯人員更改作者提交的文檔,但當生產部門將其更改內容合並到該文檔的主復本時,發現作者已將未編輯的新材料添加到該文檔中。如果在編輯人員和生產部門完成對原始文檔的處理之前,任何人都不能將新材料添加到文檔中,則可以避免該問題。

事務A connection.setAutoCommit(false) start Transaction  

update student set money=money+100

from account

where name='a'

  commit();   insert into account (name,money)values('fff',10000);    
(事務B) connection.setAutoCommit(false) start Transaction

select money from account where name='a';

發現有1000塊

 

 

 

select money from account where name='a';

發現有1100

 

select money from account where name='a';

發現有1100

  select * from account (查詢結果包含fff的信息) commit();
說明         事務B讀到了事務A未提交的數據,發生了臟讀   事務A提交后,事務B前后SELECT查詢money的值不一致,即事務B讀到了事務A,update后的數據,發生了不可重復讀   虛讀(即一個事務讀到了另一個事務insert的數據)  
可設置的事務級別         read commit   repeatable read   串行化serializable  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM