亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

詳解MySQL 幻讀及如何消除

瀏覽:3日期:2023-10-04 11:37:19

這是一篇數(shù)據(jù)庫隔離級別的科普文章,旨在了解數(shù)據(jù)庫中著名的幻讀現(xiàn)象,為了專注,對臟讀、不可重復讀不作討論。

事務隔離級別

MySQL有四級事務隔離級別:

讀未提交 READ-UNCOMMITTED: 存在臟讀,不可重復讀,幻讀的問題讀已提交 READ-COMMITTED:不存在臟讀,但存在不可重復讀,幻讀問題可重復讀 REPEATABLE-READ:不存在臟讀,不可重復讀問題,但存在幻讀問題序列化SERIALIZABLE:解決臟讀,不可重復讀,幻讀問題,但完全串行執(zhí)行,性能最低

什么是幻讀

幻讀錯誤的理解:說幻讀是事務A 執(zhí)行兩次 select 操作得到不同的數(shù)據(jù)集,即 select 1 得到10條記錄,select 2 得到11條記錄。這其實并不是幻讀,這是不可重復讀的一種,只會在 R-U R-C 級別下出現(xiàn),而在 mysql 默認的 RR 隔離級別是不會出現(xiàn)的。

這里給出我對幻讀的理解:

幻讀,并不是說事務中多次讀取獲取的結(jié)果集不同,幻讀更重要的是某次的 select 操作得到的結(jié)果集所表征的數(shù)據(jù)狀態(tài)無法支撐后續(xù)的業(yè)務操作。更為具體一些:select 記錄不存在,準備插入此記錄,但執(zhí)行 insert 時發(fā)現(xiàn)此記錄已存在,無法插入,如同產(chǎn)生了幻覺

舉個例子可能會簡化理解:

mysql> show create table userG*************************** 1. row *************************** Table: userCreate Table: CREATE TABLE `user` ( `id` int(11) NOT NULL, `name` varchar(32) DEFAULT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB DEFAULT CHARSET=utf8

分別開啟兩個事務T1 & T2,并設置其隔離級別為Reaptable-Read:

T1:

mysql> set global transaction isolation level repeatable read; ​mysql> begin;mysql> select * from user;mysql> insert into user values (1, ’jeff’);ERROR 1062 (23000): Duplicate entry ’1’ for key ’PRIMARY’​mysql> select * from user;

T2:

mysql> set global transaction isolation level repeatable read; ​mysql> begin;mysql> insert into user values (1, ’jeff’);mysql> commit;

T1 事務檢測表中是否有 id 為 1 的記錄,沒有則插入

T2 插入干擾記錄,造成T1出現(xiàn)幻讀。

上例中需要確保T1事務執(zhí)行begin后才開始執(zhí)行事務T2。

上例中T1就發(fā)生了幻讀,因為 T1讀取的數(shù)據(jù)狀態(tài)與后面的動作發(fā)生了語義上的沖突:查詢的時候明明提示記錄不存在,插入的時候去提示主鍵重復,類似于出現(xiàn)幻影,因而稱之為幻讀。

如何消除幻讀

MySQL當前有兩種方式可以消除幻讀:

1. 通過對select操作手動加行X鎖(SELECT ... FOR UPDATE )。原因是InnoDB中行鎖鎖定的是索引,縱然當前記錄不存在,當前事務也會獲得一把記錄鎖(記錄存在就加行X鎖,不存在就加next-key lock間隙X鎖),這樣其他事務則無法插入此索引的記錄,杜絕幻讀。2. 進一步提升隔離級別為SERIALIZABLE測試一下效果

mysql> begin;​mysql> select * from user where id = 2 for update;mysql> insert into user values (2, ’tony’);mysql> commit;

T2:

mysql> begin;​mysql> insert into user values (2, ’jimmy’);ERROR 1062 (23000): Duplicate entry ’2’ for key ’PRIMARY’

現(xiàn)在T1查詢時攜帶了for update,在Innodb內(nèi)會對該索引加鎖(即使當前不存在),于是事務T2的insert會被阻塞直到T1顯示提交,這樣T1成功了,對于T1來說,幻讀確實被消除了,但T2的插入會報主鍵重復,這也符合預期。

至于另外一種提升隔離級別消除幻讀的方式感興趣的可以自己嘗試,這里不再重復,其本質(zhì)是類似的,只是讓系統(tǒng)代替了手工加鎖。

總結(jié)

RR作為 mysql 事務默認隔離級別,是事務安全與性能的折中,正確認識幻讀后,開發(fā)者便可以根據(jù)需求自行決定是否需要防止幻讀。

SERIALIZABLE則是悲觀的認為幻讀時刻都會發(fā)生,故會自動的隱式的對事務所需資源加排它鎖,其他事務訪問此資源會被阻塞等待,故事務是安全的,但需要認真考慮性能。

InnoDB的鎖是針對索引,這點需要引起注意。對行記錄加鎖,如果存在,加X鎖,否則會加 next-key lock / gap 鎖 / 間隙鎖,故InnoDB可以實現(xiàn)事務對某記錄的預先占用,只要本事務還在,其他事務就別想占有它。關于鎖,后面還會再有專門的文章討論。

以上就是詳解MySQL 幻讀及如何消除的詳細內(nèi)容,更多關于MySQL 幻讀及消除的資料請關注好吧啦網(wǎng)其它相關文章!

相關文章:
主站蜘蛛池模板: 婷婷综合网站 | 在线伊人 | 国内精品小视频在线 | 国产精品视频网站你懂得 | 久久精品国产精品亚洲婷婷 | 色婷婷综合缴情综六月 | 性色综合 | 五月天婷婷在线视频 | 免费精品一区二区三区在线观看 | 国产老肥熟xxxx | 一区二区三区四区视频在线观看 | 在线高清视频观看vvvv | 国产成人一区二区三区在线播放 | 99亚洲乱人伦精品 | 国产欧美日韩看片片在线人成 | 成人午夜啪啪免费网站 | 免费人成又黄又爽的视频在线 | 激情丁香网 | 国产麻豆精品入口在线观看 | 免费a一级毛片在线播放 | 日韩欧美国产亚洲 | 国产精品大片天天看片 | 欧美日韩亚洲综合久久久 | 国产成人精品在线 | 亚洲人成网站观看在线播放 | 国产免费一区二区三区在线观看 | 手机看片www xiao2b cm | 日韩免费一区二区三区 | 国产a级男女性高爱潮片 | 久久97精品久久久久久清纯 | 欧美亚洲免费 | 成人一级黄色大片 | 欧美日韩在线免费观看 | 黄色视屏日本 | 手机看片中文字幕 | 91专区在线观看 | 亚洲第99页 | 久久三级国产 | 视频在线二区 | 正在播放国产精品放孕妇 | 国产成人污污网站在线观看 |