外來鍵欄位未建索引引發的死鎖
現象:一個很簡單的程式在壓力測試過程中發現死鎖,檢視trace檔案,發現如下資訊:
Deadlock graph:
---------Blocker(s)-------- ---------Waiter(s)---------
Resource Name process session holds waits process session holds waits
TM-0000627d-00000000 21 1062 SX SSX 22 1092 SX SSX
TM-0000627d-00000000 22 1092 SX SSX 21 1062 SX SSX
session 1062: DID 0001-0015-0000001E session 1092: DID 0001-0016-0005C757
session 1092: DID 0001-0016-0005C757 session 1062: DID 0001-0015-0000001E
Rows waited on:
Session 1092: no row
Session 1062: no row
而引發這個死鎖的sql如下:
Current SQL Statement:
DELETE FROM TABLE WHERE COL = :B1
首先注意到的是Deadlock graph中的資源佔有情況,可以看到兩個session都hold了一個SZ型別的鎖,同時在等待SSX型別的鎖,而且引發的是一個刪除語句,並且這個表是系統的一個關鍵表,大部分的表的外來鍵都引用自此表的主鍵。因此猜測是碰到了外來鍵引發的死鎖。試驗如下:
1、建立一個表,此表作為子表
SQL> create table fk_table as select * from user_objects;
Table created
2、建立一個表,此表作為父表
SQL> create table pk_table as select * from user_objects;
Table created
3、建立父表的主鍵
SQL> alter table PK_TABLE add constraint pk_pktable primary key (OBJECT_ID);
Table altered
4、建立子表的外來鍵
SQL> alter table FK_TABLE add constraint fk_fktable foreign key (OBJECT_ID) references pk_table (OBJECT_ID);
Table altered
5、如下sql取自TOAD工具,用來顯示資料庫鎖的資訊
SELECT LK.SID,
SE.USERNAME,
SE.OSUSER,
SE.MACHINE,
DECODE(LK.TYPE,
'TX',
'Transaction',
'TM',
'DML',
'UL',
'PL/SQL User Lock',
LK.TYPE) LOCK_TYPE,
DECODE(LK.LMODE,
0,
'None',
1,
'Null',
2,
'Row-S (SS)',
3,
'Row-X (SX)',
4,
'Share',
5,
'S/Row-X (SSX)',
6,
'Exclusive',
TO_CHAR(LK.LMODE)) MODE_HELD,
DECODE(LK.REQUEST,
0,
'None',
1,
'Null',
2,
'Row-S (SS)',
3,
'Row-X (SX)',
4,
'Share',
5,
'S/Row-X (SSX)',
6,
'Exclusive',
TO_CHAR(LK.REQUEST)) MODE_REQUESTED,
TO_CHAR(LK.ID1) LOCK_ID1,
TO_CHAR(LK.ID2) LOCK_ID2,
OB.OWNER,
OB.OBJECT_TYPE,
OB.OBJECT_NAME,
LK.BLOCK,
SE.LOCKWAIT
FROM V$LOCK LK, DBA_OBJECTS OB, V$SESSION SE
WHERE LK.TYPE IN ('TM', 'UL')
AND LK.SID = SE.SID
AND LK.ID1 = OB.OBJECT_ID(+);
6、執行一個刪除操作,這時候在子表和父表上都加了一個Row-S(SX)鎖
delete from fk_table where object_id=94716;
delete from pk_table where object_id=94716;
7、執行另一個刪除操作,發現這時候第二個刪除語句等待
delete from fk_table where object_id=94702;
delete from pk_table where object_id=94702;
執行查詢語句,得到鎖資訊如下:
857 DML Row-S (SS) None 107220 0 BILL TABLE PK_TABLE 0 00000000D555A0E8
872 DML Row-X (SX) None 107220 0 BILL TABLE PK_TABLE 0
857 DML Row-X (SX) S/Row-X (SSX) 107219 0 BILL TABLE FK_TABLE 0 00000000D555A0E8
872 DML Row-X (SX) None 107219 0 BILL TABLE FK_TABLE 1
可以看到session 857在請求一個SSX型別的鎖,此時如果執行如下操作:
delete from pk_table where object_id=94716;
死鎖馬上發生,因為857 SESSION拿到了一個對FK_TABLE的行獨佔鎖,並在請求一個表共享鎖,而872 SESSION也拿到了一個FK_TABLE上的行獨佔鎖,並請求一個表共享鎖。此時兩個session誰都不會釋放獨佔鎖,並同時請求表的共享鎖,死鎖由此引發。因為死鎖引發的時候兩個session不是在等待對資料行進行加鎖,所以可以從trace檔案中發現等待的行都為no row,同時可以看到兩個session都hold了一個SX鎖,並且都在等待SSX鎖資源。同時trace檔案中還記錄了引發死鎖的sql。
Deadlock graph:
---------Blocker(s)-------- ---------Waiter(s)---------
Resource Name process session holds waits process session holds waits
TM-0001a2d3-00000000 16 872 SX SSX 20 857 SX SSX
TM-0001a2d3-00000000 20 857 SX SSX 16 872 SX SSX
session 872: DID 0001-0010-000F5EA0 session 857: DID 0001-0014-001D7407
session 857: DID 0001-0014-001D7407 session 872: DID 0001-0010-000F5EA0
Rows waited on:
Session 857: no row
Session 872: no row
Current SQL Statement:
delete from pk_table where object_id=94716
8、當對子表的外來鍵列新增索引後,死鎖被消除,因為這時刪除父表記錄不需要對子表加表級鎖,這裡不再做測試。
結論:曾經有人討論過是否所有的資料庫設計都應該遵守正規化的規範,都把主外來鍵關係建立起來。也有人反對這樣做,因為這樣複雜的關係在OLTP系統中可能會成為災難,而提倡透過程式來保證資料的完整性,但程式發生bug導致資料不一致的情況時有發生。而且如果外來鍵設定為級聯刪除,則不加索引的外來鍵會使得對子表的記錄刪除走全表掃描。因此,對外來鍵的使用還是要慎重!
btw:為什麼刪除子表記錄的時候要加表級的共享鎖呢?還有這個加鎖好像只是一瞬間的,期待深入!!!
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/25016/viewspace-923330/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 主鍵衝突引發的死鎖
- 10.30 索引,外來鍵索引
- 如果新增欄位是外來鍵,如何處理?
- SQLServer索引優化(1):對於有order by欄位的建索引策略SQLServer索引優化
- enq: TM - contention解決之道——外來鍵無索引導致鎖爭用ENQ索引
- SQLSERVER 語句交錯引發的死鎖研究SQLServer
- openGauss-外來鍵鎖增強
- MySQL:一個死鎖分析 (未分析出來的死鎖)MySql
- 不合理的執行順序引發的死鎖
- [20181020]lob欄位的索引段.txt索引
- 億級大表線上不鎖表變更欄位與索引索引
- oracle複合索引介紹(多欄位索引)Oracle索引
- 【Django drf】 序列化類常用欄位類和欄位引數 定製序列化欄位的兩種方式 關係表外來鍵欄位的反序列化儲存 序列化類繼承ModelSerializer 反序列化資料校驗原始碼分析Django繼承原始碼
- 併發:死鎖
- PG裡常見的欄位有索引但未使用索引的原因索引
- 外來鍵沒有索引哪些DML操作會被阻塞索引
- insert引起的死鎖,起因主鍵衝突
- 怎麼給字串欄位加索引?字串索引
- redis對hash欄位加鎖Redis
- MYSQL的外來鍵MySql
- CSV中欄位的去掉雙引號
- Django 常用欄位和引數Django
- WTForms 欄位 引數 校驗ORM
- MySQL null值欄位是否使用索引的總結MySqlNull索引
- 無索引的外來鍵之主表子表DML操作實驗及結論索引
- [BUG反饋]模型增加欄位後field_sort欄位未更新模型
- 作業系統(5) 死鎖的概念 死鎖產生的必要條件 死鎖的處理策略 預防死鎖 避免死鎖 死鎖的檢測和解除 銀行家演算法作業系統演算法
- EBS:主鍵ID欄位預設值來源於序號
- RPA和人工智慧:解鎖未來企業制勝的關鍵人工智慧
- [Java併發]避免死鎖Java
- 外來鍵的變種
- 一次mongo查詢不存在欄位引發的事故Go
- 主鍵和外來鍵
- mysqldump 欄位值帶單引號MySql
- [20190810]如何索引一個超長欄位.txt索引
- sqlserver外來鍵SQLServer
- 死鎖
- [20180408]那些函式索引適合欄位的查詢.txt函式索引