更新時(shí)間:2022-11-18 來源:黑馬程序員 瀏覽量:
一、什么是死鎖
加鎖(Locking)是數(shù)據(jù)庫在并發(fā)訪問時(shí)保證數(shù)據(jù)一致性和完整性的主要機(jī)制。任何事務(wù)都需要獲得相應(yīng)對(duì)象上的鎖才能訪問數(shù)據(jù),讀取數(shù)據(jù)的事務(wù)通常只需要獲得讀鎖(共享鎖),修改數(shù)據(jù)的事務(wù)需要獲得寫鎖(排他鎖)。當(dāng)兩個(gè)事務(wù)互相之間需要等待對(duì)方釋放獲得的資源時(shí),如果系統(tǒng)不進(jìn)行干預(yù)則會(huì)一直等待下去,也就是進(jìn)入了死鎖(deadlock)狀態(tài)。
二、死鎖產(chǎn)生的場(chǎng)景
1.數(shù)據(jù)庫表準(zhǔn)備
1.1. 創(chuàng)建數(shù)據(jù)庫表
#創(chuàng)建數(shù)據(jù)庫 create database chuanzhi; #選擇數(shù)據(jù)庫 use chuanzhi; #創(chuàng)建測(cè)試表 CREATE TABLE `tb_heima` ( `id` INT NOT NULL, `username` VARCHAR(45) NULL, PRIMARY KEY (`id`) ); #插入測(cè)試數(shù)據(jù) INSERT INTO `tb_heima` (`id`, `username`) VALUES ('1', '傳智教育-教育行業(yè)A股IPO第一股');
2.死鎖場(chǎng)景演示
2.1. 打開一個(gè)dos窗口
- mysql -u賬號(hào) -p密碼,進(jìn)入mysql命令行輸入:
- 開啟一個(gè)事務(wù),mysql> begin;
- 執(zhí)行更新語句,mysql> update tb_heima set username = 'heima' where id =1;
注意:此時(shí)更新是成功的,但并未提交事務(wù)。
2.2.再打開一個(gè)dos窗口
- 登錄mysql,開啟另一個(gè)事務(wù)操作同上。
- 再次執(zhí)行更新語句,mysql> update tb_heima set username = 'heima' where id =1;
注意:此時(shí)第二個(gè)窗口處于等待中,需要等待第一個(gè)窗口釋放寫鎖才能執(zhí)行成功。
- 等待一段時(shí)候后,第二個(gè)窗口報(bào)錯(cuò)了
注意:由于第一個(gè)窗口未提交事務(wù),導(dǎo)致第二個(gè)窗口在等待一段時(shí)候后,超過鎖定等待超時(shí)。
三、解決死鎖問題
通過mysql客戶端工具連接,按照以下步驟執(zhí)行。
- select from information_schema.innodb_lock_waits; 鎖等待的信息,可以看到堵塞和被堵塞者
注意:超過鎖定等待超時(shí)后,此條記錄會(huì)消失,也就是說正在阻塞的記錄在此表才能查看到。
- select * from information_schema.innodb_trx; 查詢鎖事務(wù)狀態(tài)信息
注意:查詢出死鎖后,得到trx_mysql_thread_id死鎖的id。
- kill sessionId; 殺掉有問題的session
kill 15; kill 16;
- 再次執(zhí)行select * from information_schema.innodb_trx; # 查詢鎖事務(wù)狀態(tài)信息
注意:確認(rèn)此表是否存在鎖事務(wù)記錄,如果沒有則說明死鎖問題已經(jīng)解決。
四、如何避免死鎖
1.對(duì)于數(shù)據(jù)庫的多表操作時(shí),盡量按照相同的順序進(jìn)行處理,盡量避免同時(shí)鎖定兩個(gè)資源,如操作A和B兩張表時(shí),總是按先A后B的順序處理, 必須同時(shí)鎖定兩個(gè)資源時(shí),要保證在任何時(shí)刻都應(yīng)該按照相同的順序來鎖定資源。
2. 所有的update和delete操作必須走唯一索引
3. SQL語句中不要使用太復(fù)雜的關(guān)聯(lián)多表的查詢;使用“執(zhí)行計(jì)劃”對(duì)SQL語句進(jìn)行分析,對(duì)于有全表掃描的SQL語句,建立相應(yīng)的索引進(jìn)行優(yōu)化。
4. 把SELECT放在Update語句前
5. 避免事務(wù)中的用戶等待交互