mysql for update是鎖表還是鎖行實例詳解
在并發(fā)一致性控制場景中,我們常常用for update
悲觀鎖來進行一致性的保證,但是如果不了解它的機制,就進行使用,很容易出現(xiàn)事故,比如for update
進行了鎖表導(dǎo)致其他請求只能等待,從而拖垮系統(tǒng),因此了解它的原理是非常必要的,下面我們通過一系列示例進行測試,來看看到底是什么場景下鎖表什么場景下鎖行
驗證
示例說明
創(chuàng)建一個賬戶表,插入基礎(chǔ)數(shù)據(jù),以唯一索引
、普通索引
、主鍵
、普通字段
4 個維度進行select ... for update
查詢,查看是進行鎖表還是鎖行
表創(chuàng)建
創(chuàng)建一個賬戶表,指定account_no
為唯一索引、id
為主鍵、user_no
為普通字段、curreny
為普通索引
CREATE TABLE `account_info` ( `id` int NOT NULL AUTO_INCREMENT COMMENT 'ID' , `account_no` int NOT NULL COMMENT '賬戶編號', `user_no` varchar(32) NOT NULL COMMENT '用戶 Id', `currency` varchar(10) NOT NULL COMMENT '幣種', `amount` DECIMAL(10,2) NOT NULL COMMENT '金額', `freeze_amount` DECIMAL(10,2) NOT NULL COMMENT '凍結(jié)金額', `create_time` datetime(6) NOT NULL DEFAULT CURRENT_TIMESTAMP(6) COMMENT '創(chuàng)建時間', `update_time` datetime(6) NOT NULL DEFAULT CURRENT_TIMESTAMP(6) ON UPDATE CURRENT_TIMESTAMP(6) COMMENT '修改時間', PRIMARY KEY (`id`) USING BTREE, UNIQUE KEY `uni_idx_account_no` (`account_no`) , KEY `idx_currency_` (`currency`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='賬戶信息表';
插入基礎(chǔ)數(shù)據(jù)
insert into account_info values (1,1,'ur001','RMB',100,0,now(),now()); insert into account_info values (2,2,'ur002','RMB',1000,0,now(),now()); insert into account_info values (3,3,'ur002','DOLLAR',200,0,now(),now());
根據(jù)主鍵查詢
在事務(wù) 1 中,根據(jù)主鍵id=1 進行 for update查詢時,事務(wù)2、事務(wù) 3 都進行阻塞,而事務(wù) 4 由于更新的id=2 所以成功,因此判定,根據(jù)主鍵進行 for update 查詢時是行鎖
根據(jù)唯一索引查詢
在事務(wù) 1 中,根據(jù)唯一索引account_no=1 進行 for update查詢時,事務(wù)2、事務(wù) 3 都進行阻塞,而事務(wù) 4 由于更新的account_no=2 所以成功,因此判定,根據(jù)唯一索引進行 for update 查詢時是行鎖
根據(jù)普通索引查詢
在事務(wù) 1 中,根據(jù)普通索引currency='RMB' 進行 for update查詢時,事務(wù)2、事務(wù) 3 都進行阻塞,而事務(wù) 4 由于更新的currency='DOLLAR`所以成功,因此判定,根據(jù)普通索引進行 for update 查詢時是行鎖
根據(jù)普通字段查詢
在事務(wù) 1 中,根據(jù)普通字段user_no='ur001' 進行 for update查詢時,事務(wù)2、事務(wù) 3 都進行阻塞,而事務(wù) 4查詢的是user_no='ur002'也進行阻塞,因此判定,根據(jù)普通字段進行 for update 查詢時是表鎖
總結(jié)
如果查詢條件是索引/主鍵字段,那么select ..... for update
會進行行鎖
如果查詢條件是普通字段(沒有索引/主鍵),那么select ..... for update
會進行鎖表,這點一定要注意。
掃描下面的二維碼關(guān)注我們的微信公眾帳號,在微信公眾號中共同學(xué)習(xí)。
到此這篇關(guān)于mysql for update是鎖表還是鎖行實例詳解的文章就介紹到這了,更多相關(guān)mysql for update內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決mysql登錄錯誤:''Access denied for user ''root''@''localhost''
這篇文章主要介紹了mysql登錄錯誤:'Access denied for user 'root'@'localhost',本文給出了操作過程及注意事項,需要的朋友可以參考下2019-11-11MySQL使用Sequence創(chuàng)建唯一主鍵的實現(xiàn)示例
Sequence提供了更多的靈活性,本文主要介紹了MySQL使用Sequence創(chuàng)建唯一主鍵的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-05-05