幻讀的定義是指,一個事務開啟后,執行前后兩次查詢,兩次查詢中出現了新的數據,幻讀僅針對數據的新增。
創新互聯建站憑借在網站建設、網站推廣領域領先的技術能力和多年的行業經驗,為客戶提供超值的營銷型網站建設服務,我們始終認為:好的營銷型網站就是好的業務員。我們已成功為企業單位、個人等客戶提供了成都網站制作、成都網站設計服務,以良好的商業信譽,完善的服務及深厚的技術力量處于同行領先地位。
比如: 表t中,id為主鍵,目前有數據1,5,10,20四條。
開始一個事務A,前后兩次執行 select * from t where id 10 for update;
開啟一個事務B,在事務A第二次執行查詢前,執行insert into t values( 2,...); 并提交事務(請暫時忽略這里能否成功執行!)。
此時在RC、RR隔離級別下都會導致事務A第二次查詢能夠查詢到 事務B新增的數據 id = 2。
RC級別下能夠看到不同結果就不做解釋了。
對于RR隔離級別下,有了MVCC版本控制為什么還能讀取到不同的結果呢?
這里要歸功于 "for update"。
"for update" 會將快照讀變為當前讀,在當前讀場景中,會自動讀取最新的數據,而非快照數據。
分析一下,鎖與當前讀關系。了解什么情況下會加鎖。了解 意向鎖、共享鎖、排它鎖區別及各自在什么情況下使用。
行鎖的概念都清楚,這里就不做補充了。
間隙鎖實際上是指一個區間。
我們都知道,InnoDB 在RR事務隔離級別下解決幻讀問題就是通過Next Key Lock (間隙鎖+行鎖)來實現的。而且,很多地方也有提到,如果對于讀一致性要求不高的場景可以考慮使用RC隔離級別,允許幻讀的發生。
還是上邊說的那個實例,略微改動:
比如: 表t中,id為主鍵,目前有數據1,5,10,20四條。
開始一個事務A,前后三次分別執行
開啟一個事務B,在事務A執行update前,執行insert into t values( 2,...); 并提交事務。
此時我們知道,事務A中第二次查詢能夠查到 事務B新增的數據,也就是產生了幻讀。那么,按照SQL執行的順序來說,事務B
幻讀是指:在一個事務中,讀取到了其他已經提交的事務插入的數據行。
MySQL在解決臟讀、不可重復的讀時候,使用了MVCC一致性視圖,同時配合行鎖來解決。
至于幻讀的解決方式,MySQL引入了臨鍵鎖,通過間隙鎖可以避免在兩個行之間插入數據,從而避免了一個事務在讀取的過程中,讀取到其他事務插入的數據行。
什么是幻讀?
幻讀指的是一個事務在前后兩次查詢同一個范圍的時候,后一次查詢看到了前一次查詢沒有看到的行。
首先快照讀是不存在幻讀的,只有當前讀(實時讀)才存在幻讀的問題。
幻讀有什么問題?
select ...for update語句就是將相應的數據行鎖住,但是如果存在幻讀,就把for update的語義破壞了。
如何解決幻讀?
產生幻讀的原因是,行鎖只能鎖住行,但是新插入記錄這個動作,要更新的是記錄之間的“間隙”。因此,為了解決幻讀問題,InnoDB只好引入新的鎖,也就是間隙鎖(Gap Lock)。間隙鎖和行鎖合稱 next-key lock , 每個next-key lock是前開后閉區間 。
總結
一、定義
1、幻讀MYSQL官方叫法是Phantom Rows,意為鬼影行或者幻影行,請看官方定義:
The so-called phantom problem occurs within a transaction when the same query produces different sets of rows at different times. For example, if a [ SELECT ] is executed twice, but returns a row the second time that was not returned the first time, the row is a “phantom” row.
翻譯一下:
所謂的幻影行問題是指,在同一個事務中,同樣的查詢語句執行多次,得到了不同的行結果集。
例如,如果同一個SELECT語句執行了兩次,第二次執行的時候比第一次執行時多出一行,則該行就是所謂的幻影行。
2、幻讀與不可重復讀的區別
從官方的定義來看,幻讀的定義側重于多條記錄,就是記錄條數的變化,而不可重復讀側重于單條記錄數據的變化,這樣區分原因在于解決幻讀需要范圍鎖,解決不可重復讀只需要單條記錄加鎖
二、InnoDB的REPEATABLE READ級別
InnoDB支持由SQL1992標準描述的所有四個事務隔離級別,默認隔離級別是 REPEATABLE READ。
1、快照讀:
在RR模式下,第一次讀取會建立快照,后續查詢會讀取快照。
這意味著,如果在同一事務中發出多個普通[ SELECT ](非鎖定)語句,則這些 [ SELECT ]語句的結果也是一致的。
2、[locking reads](鎖定讀取,又叫當前讀)
[ SELECT ]語句中使用 FOR UPDATE 或 FOR SHARE
3、行鎖
在RR模式下,使用當前讀以及 [ UPDATE ]和 [ DELETE ]語句會對數據記錄加行鎖,鎖定范圍取決于該語句使用的是具有唯一搜索條件的唯一索引還是范圍類型搜索條件。
三、InnoDB的READ COMMITTED級別
1、在RC模式下,每次讀取都會刷新快照,因此不能保證可重復讀
2、在RC模式下,使用當前讀以及 [ UPDATE ]和 [ DELETE ]語句會對數據記錄加行鎖,但是不會加范圍鎖,間隙鎖定僅用于外鍵約束檢查和重復鍵檢查。
3、由于禁用了間隙鎖定,因此可能會產生幻影行問題,因為其他會話可以在間隙中插入新行。
4、 對于[ UPDATE ]或 [ DELETE ]語句, InnoDB 僅對其更新或刪除的行持有鎖。MySQL評估 WHERE 條件后,將釋放不匹配行的記錄鎖 。這大大降低了死鎖的可能性,但是仍然可以發生。
5、對于[ UPDATE ]語句,如果某行已被鎖定,則 InnoDB 執行“半一致”讀取,將最新提交版本的數據返回給MySQL,以便MySQL可以確定該行是否符合 WHERE 條件。如果該行匹配(必須更新),則MySQL會再次讀取該行,這一次 InnoDB 會將其鎖定或等待獲取鎖。
6、注意
從MySQL 8.0.22開始,DML操作(增刪改,通過聯接列表或子查詢)從MySQL授權表中讀取數據,但不對其進行修改,無論隔離級別如何,都不會在MySQL授權表上獲得讀取鎖。
有關更多信息,請參見 Grant Table Concurrency 。
四、樂觀鎖與悲觀鎖
1、樂觀鎖
在UPDATE的WHERE子句中加入版本信息來確定修改是否生效
使用樂觀鎖時仍然需要非常謹慎,因為RR是可重復讀的,在UPDATE之前讀取版本號,應該使用[當前讀],不能使用[快照讀]
2、悲觀鎖
在UPDATE執行前,SELECT后面加上FOR UPDATE來給記錄加鎖,保證記錄在UPDATE前不被修改。SELECT ... FOR UPDATE是加上了X鎖,也可以通過SELECT ... LOCK IN SHARE MODE加上S鎖,來防止其他事務對該行的修改。
3、無論是樂觀鎖還是悲觀鎖,使用的思想都是一致的,那就是當前讀。樂觀鎖利用當前讀判斷是否是最新版本,悲觀鎖利用當前讀鎖定行。
五、總結
1、RC級別沒有范圍鎖一定會導致不可重復讀和幻影行
2、RR級別安全性更高,實現可重復讀的方式為快照,如果需要最新數據可以選擇[當前讀],因此RR級別是首選
3、不論RR還是RC級別,增、刪、改的操作都會進行一次[當前讀]操作,以此獲取最新版本的數據,并檢測是否有重復的索引。
4、RR級別下,當前事務如果未發生更新操作(增刪改),快照版本會保持不變,多次查詢讀取的快照是同一個
5、RR級別下,當前事務如果發生更新(增刪改),會刷新快照,會導致不可重復讀和幻影行
6、RR級別下,使用當前讀,會刷新快照,會導致不可重復讀和幻影行
7、RR級別下,可以通過提交當前事務并在此之后發出新查詢來為查詢獲取更新的快照。
8、RR級別可以部分解決不可重復讀和幻讀問題
9、其實問題的關鍵是你的業務邏輯需要可重復讀還是最新數據
我們設想一個場景,這個場景中我們需要插入多條相關聯的數據到數據庫,不幸的是,這個過程可能會遇到下面這些問題:
上面的任何一個問題都可能會導致數據的不一致性。為了保證數據的一致性,系統必須能夠處理這些問題。事務就是我們抽象出來簡化這些問題的首選機制。事務的概念起源于數據庫,目前,已經成為一個比較廣泛的概念。
何為事務? 一言蔽之, 事務是邏輯上的一組操作,要么都執行,要么都不執行。
事務最經典也經常被拿出來說例子就是轉賬了。假如小明要給小紅轉賬 1000 元,這個轉賬會涉及到兩個關鍵操作,這兩個操作必須都成功或者都失敗。
事務會把這兩個操作就可以看成邏輯上的一個整體,這個整體包含的操作要么都成功,要么都要失敗。這樣就不會出現小明余額減少而小紅的余額卻并沒有增加的情況。
大多數情況下,我們在談論事務的時候,如果沒有特指 分布式事務 ,往往指的就是 數據庫事務 。
數據庫事務在我們日常開發中接觸的最多了。如果你的項目屬于單體架構的話,你接觸到的往往就是數據庫事務了。
那數據庫事務有什么作用呢?
簡單來說,數據庫事務可以保證多個對數據庫的操作(也就是 SQL 語句)構成一個邏輯上的整體。構成這個邏輯上的整體的這些數據庫操作遵循: 要么全部執行成功,要么全部不執行 。
另外,關系型數據庫(例如: MySQL 、 SQL Server 、 Oracle 等)事務都有 ACID 特性:
ACID
這里要額外補充一點: 只有保證了事務的持久性、原子性、隔離性之后,一致性才能得到保障。也就是說 A、I、D 是手段,C 是目的!
在典型的應用程序中,多個事務并發運行,經常會操作相同的數據來完成各自的任務(多個用戶對同一數據進行操作)。并發雖然是必須的,但可能會導致以下的問題。
不可重復讀和幻讀區別 :不可重復讀的重點是修改比如多次讀取一條記錄發現其中某些列的值被修改,幻讀的重點在于新增或者刪除比如多次查詢同一條查詢語句(DQL)時,記錄發現記錄增多或減少了。
SQL 標準定義了四個隔離級別:
隔離級別臟讀不可重復讀幻讀 READ-UNCOMMITTED READ-COMMITTED REPEATABLE-READ SERIALIZABLE
MySQL 的隔離級別基于鎖和 MVCC 機制共同實現的。
SERIALIZABLE 隔離級別,是通過鎖來實現的。除了 SERIALIZABLE 隔離級別,其他的隔離級別都是基于 MVCC 實現。
不過, SERIALIZABLE 之外的其他隔離級別可能也需要用到鎖機制,就比如 REPEATABLE-READ 在當前讀情況下需要使用加鎖讀來保證不會出現幻讀。
MySQL InnoDB 存儲引擎的默認支持的隔離級別是 REPEATABLE-READ(可重讀) 。我們可以通過 SELECT @@tx_isolation; 命令來查看,MySQL 8.0 該命令改為 SELECT @@transaction_isolation;
從上面對 SQL 標準定義了四個隔離級別的介紹可以看出,標準的 SQL 隔離級別定義里,REPEATABLE-READ(可重復讀)是不可以防止幻讀的。
但是!InnoDB 實現的 REPEATABLE-READ 隔離級別其實是可以解決幻讀問題發生的,主要有下面兩種情況:
因為隔離級別越低,事務請求的鎖越少,所以大部分數據庫系統的隔離級別都是 READ-COMMITTED ,但是你要知道的是 InnoDB 存儲引擎默認使用 REPEATABLE-READ 并不會有任何性能損失。
InnoDB 存儲引擎在分布式事務的情況下一般會用到 SERIALIZABLE 隔離級別。
網站標題:mysql怎么消除幻讀,mysql怎么避免幻讀
網站網址:http://vcdvsql.cn/article0/dsigpio.html
成都網站建設公司_創新互聯,為您提供云服務器、軟件開發、網站建設、外貿建站、全網營銷推廣、定制開發
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯