在MySQL中,如果一個事務(wù)讀取了由另一個事務(wù)正在修改的數(shù)據(jù),而且這個數(shù)據(jù)還沒有被提交,那么就會產(chǎn)生臟讀。這會導(dǎo)致事務(wù)讀取到錯誤的數(shù)據(jù),從而產(chǎn)生錯誤的結(jié)果。怎么解決臟讀呢?這里我們來看看MySQL是如何解決臟讀問題的。
首先,我們需要了解一下MySQL的四個事務(wù)隔離級別:讀未提交、讀已提交、可重復(fù)讀和串行化。
-- 設(shè)置事務(wù)隔離級別為可重復(fù)讀
SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;
-- 開啟事務(wù)
START TRANSACTION;
-- 執(zhí)行一系列的SQL操作
-- ...
-- 提交事務(wù)
COMMIT;
如果我們采用默認(rèn)的隔離級別(讀已提交),那么MySQL是無法避免臟讀的。因為讀已提交級別下,當(dāng)一個事務(wù)正在修改某個數(shù)據(jù)時,讀取該數(shù)據(jù)的其他事務(wù)會等待該事務(wù)提交后才能讀取。但是,在該事務(wù)提交之前,其他事務(wù)可以讀取該數(shù)據(jù),從而產(chǎn)生臟讀。
如果要避免臟讀,我們可以將隔離級別設(shè)置為可重復(fù)讀,這是MySQL默認(rèn)的事務(wù)隔離級別。在可重復(fù)讀級別下,一旦一個事務(wù)讀取了某個數(shù)據(jù),那么在該事務(wù)提交前,其他事務(wù)只能讀取該數(shù)據(jù)的一個快照,而不是直接讀取。這樣可以避免臟讀。
當(dāng)然,如果對數(shù)據(jù)的訪問需求非常嚴(yán)格,我們可以將隔離級別設(shè)置為串行化。在串行化級別下,事務(wù)之間的操作是完全隔離的,事務(wù)的執(zhí)行是完全串行的,這樣可以避免所有的并發(fā)問題,包括臟讀、不可重復(fù)讀和幻讀。