enq: TM - contention解決之道——外來鍵無索引導致鎖爭用(下)
上篇文章簡要介紹了一下當外來鍵無索引時,更新刪除主表的資料會造成子表的鎖定,如果此時子表上有事務,那麼進行更新刪除的session變會等待,等待事件就是enq: TM - contention
外來鍵與 TM enqueue lock 的主要問題是 在早期版本中(9i之前) 當 子表child table上 的外來鍵沒有索引時 , 若發生 父表 parent table 上記錄被delete 或 update時 , 會在child table上加 share lock, 這會 阻塞 child table 上的DML。
但是從 9i以後的當 子表child table上 的外來鍵沒有索引時, 父表parent table上的delete 、update 只在 實際這個DML執行的過程中要求share (TM lmode=4) lock,而不會在整個事務中 都要求保持 child table上的 share lock。
還是先了解一下oracle中的鎖模式吧,TM鎖和TX鎖都屬於DML鎖,這裡介紹的是TM的鎖模式
TM 鎖在下列場景中被申請:
好了,開始動手做個試驗吧,試驗中我會引用KST trace的內容,關於KST,本文不做介紹,只拿來使用
首先,準備環境,本實驗均在11.2.0.4環境下
這裡要說一下,在外來鍵是否存在on delete cascade時鎖的獲取還有區別,所以我們分別來測試,首先是沒有索引沒有cascade的情況下,各個語句的鎖獲取情況
一、無索引,無cascade
insert 父表:
檢視kst資訊
可見父表上的插入會獲取父表和子表mode為3的TM鎖,TM後跟的是object_id的十六進位制,一個TX鎖,讓我們驗證一下
我們來commit一下
可見獲得的鎖全部一一釋放
insert子表:
可見子表上的插入也會獲取父表和子表mode為3的TM鎖
update父表:
可以很清楚的看到在執行語句期間,注意僅僅是語句的執行期間,會附加一個mode為4的S鎖到子表上,很快便釋放了
delete父表:
delete跟update比多了一次S鎖的獲取和釋放,為何呢,是否和刪除的行數有關?我們再多刪一行試試
可以發現除了語句執行時需要獲取一次S鎖之外,刪多少行就要獲取多少次S鎖,從之前的鎖相容列表就可發現S鎖和SX(RX)鎖是不相容的,而SX(RX)是insert update delete獲取的鎖模式,可以想象如果此時子表上有事務,或者S鎖獲得了尚未釋放的時候,子表要進行事務獲取mode為3的SX(RX)鎖時,session都會產生等待。
看一下此時session獲取的鎖,記住這次結果,後面會有對比。
可見語句執行完,已不持有子表上的任何鎖
下面來模擬一下等待,
此時查一下等待鏈
從上面的分析我們知道,無論插入父表和子表,都會獲取兩張表上的mode為3的鎖,而mode為3的鎖和mode為4的鎖是不相容的,也就是說此時父表上連插入都無法進行
再開第三個session
檢視等待鏈
二、無索引,有cascade
有cascade的時候,僅在delete語句上有所區別,下面僅列出delete語句
此時會申請一個mode為5的SSX鎖,隨後即轉換為mode為3的SX鎖,這也是在語句執行期間獲取和轉換的,並非事務期間,同樣刪除多少行就涉及到多少次獲取轉換,看一下此時鎖獲得情況
是不是和沒有cascade的時候不同了,這次最終會持有子表上的mode為3的鎖,我們再深入的思考一點,SSX鎖和SX鎖是不相容的,這樣是否就意味著後進行的delete會被先進行的delete阻塞(不同session),好,現在就來模擬一下:
查詢此刻鎖的持有情況
可見1167在請求mode為5的鎖,且已被阻塞
檢視等待鏈
因為delete完畢會持有子表上的SX鎖,而SX鎖與S鎖不相容,所以delete父表的session也會阻塞update父表的session,因為update回去請求子表的S鎖,而此時子表上有SX鎖,類似於子表上有事務在進行,這裡就不在論述了,徒佔篇幅。
三、有索引,無cascade
我看到有資料說,如果有索引時,對父表的操作,會級聯加一個TM RS鎖(level 2)到子表上。但我在試驗中並未看到,也許是版本差異,我也未去求證,有索引時insert與無索引時在獲取鎖方面沒有區別,這裡僅列出update和delete
建立索引:
update父表:
可見有了索引之後,不再需要在語句級別獲取子表上的S鎖了
delete父表:
與update相同,都持有了子表上的SX鎖,而SX與SX是相容的,所以不會再產生鎖定問題
四、有索引,有cascade
表現與無cascade時相同
五、結論
參考文獻:
http://www.askmaclean.com/archives/know-more-about-oracle-row-lock.html
http://www.askmaclean.com/archives/enq-tm-contention.html
http://www.askmaclean.com/archives/know-oracle-lock-mode.html
外來鍵與 TM enqueue lock 的主要問題是 在早期版本中(9i之前) 當 子表child table上 的外來鍵沒有索引時 , 若發生 父表 parent table 上記錄被delete 或 update時 , 會在child table上加 share lock, 這會 阻塞 child table 上的DML。
但是從 9i以後的當 子表child table上 的外來鍵沒有索引時, 父表parent table上的delete 、update 只在 實際這個DML執行的過程中要求share (TM lmode=4) lock,而不會在整個事務中 都要求保持 child table上的 share lock。
還是先了解一下oracle中的鎖模式吧,TM鎖和TX鎖都屬於DML鎖,這裡介紹的是TM的鎖模式
Value Name(s) Table method (TM lock) 0 No lock n/a 1 Null lock (NL) Used during some parallel DML operations (e.g. update) by the pX slaves while the QC is holding an exclusive lock. 2 Sub-share (SS) Until 9.2.0.5/6 "select for update" Row-share (RS) Since 9.2.0.1/2 used at opposite end of RI during DML Lock table in row share mode Lock table in share update mode 3 Sub-exclusive(SX) Update (also "select for update" from 9.2.0.5/6) Row-exclusive(RX) Lock table in row exclusive mode Since 11.1 used at opposite end of RI during DML 4 Share (S) Lock table in share mode Can appear during parallel DML with id2 = 1, in the PX slave sessions Common symptom of "foreign key locking" (missing index) problem 5 share sub exclusive (SSX) Lock table in share row exclusive mode share row exclusive (SRX) Less common symptom of "foreign key locking" but likely to be more frequent if the FK constraint is defined with "on delete cascade." 6 Exclusive (X) Lock table in exclusive modeshare lock就是mode為4的S鎖
Summary of Locks Obtained by DML Statements
SQL Statement | Row Locks | Table Lock Mode | RS | RX | S | SRX | X |
---|---|---|---|---|---|---|---|
SELECT … FROM table... | — | none | Y | Y | Y | Y | Y |
INSERT INTO table … | Yes | SX | Y | Y | N | N | N |
UPDATE table … | Yes | SX | Y* | Y* | N | N | N |
MERGE INTO table … | Yes | SX | Y | Y | N | N | N |
DELETE FROM table … | Yes | SX | Y* | Y* | N | N | N |
SELECT … FROM table FOR UPDATE OF … | Yes | SX | Y* | Y* | N | N | N |
LOCK TABLE table IN … | — | ||||||
ROW SHARE MODE | SS | Y | Y | Y | Y | N | |
ROW EXCLUSIVE MODE | SX | Y | Y | N | N | N | |
SHARE MODE | S | Y | N | Y | N | N | |
SHARE ROW EXCLUSIVE MODE | SSX | Y | N | N | N | N | |
EXCLUSIVE MODE | X | N | N | N | N | N | |
* Yes, if no conflicting row locks are held by another transaction. Otherwise, waits occur. |
TM 鎖在下列場景中被申請:
- 在OPS(早期的RAC)中LGWR會以ID1=0 & ID2=0去申請該佇列鎖來檢查 DML_LOCKS 在所有例項中是全0還是全非0
- 當一個單表或分割槽 需要做不同的表/分割槽操作時,ORACLE需要協調這些操作,所以需要申請該佇列鎖。包括:
- 啟用參考約束 referential constraints
- 修改約束從DIASABLE NOVALIDATE 到DISABLE VALIDATE
- 重建IOT
- 建立檢視或者修改ALTER檢視時可能需要申請該佇列鎖
- 分析表統計資訊或validate structure時
- 一些PDML並行DML操作
- 所有可能呼叫kkdllk()函式的操作
- 太多太多了。。。
好了,開始動手做個試驗吧,試驗中我會引用KST trace的內容,關於KST,本文不做介紹,只拿來使用
首先,準備環境,本實驗均在11.2.0.4環境下
-
SQL> conn lp/lp
-
Connected.
-
SQL> create table prim(a int,b varchar2(10));
-
-
Table created.
-
-
SQL> alter table prim add constraint PK_PRIM primary key(a);
-
-
Table altered.
-
-
SQL> create table child (ca int,cb varchar2(10));
-
-
Table created.
-
-
SQL> alter table child add constraint FK_CHILD_CA foreign key (ca) references prim(a);
-
-
Table altered.
-
-
SQL> insert into prim values(1,'asdasd');
-
-
1 row created.
-
-
SQL> insert into prim values(2,'asdasd');
-
-
1 row created.
-
-
SQL> insert into prim values(3,'asdasd');
-
-
1 row created.
-
-
SQL> commit;
-
- Commit complete.
這裡要說一下,在外來鍵是否存在on delete cascade時鎖的獲取還有區別,所以我們分別來測試,首先是沒有索引沒有cascade的情況下,各個語句的鎖獲取情況
一、無索引,無cascade
-
SQL> select distinct sid from v$mystat;
-
-
SID
-
----------
-
17
-
-
SQL> select pid,spid from v$process where addr = ( select paddr from v$session where sid=(select distinct sid from v$mystat));
-
-
PID SPID
-
---------- ------------
-
36 2761
-
-
SQL> alter system set "_trace_events"='10000-10999:255:36';
-
- System altered.
- insert into prim values(5,'asdasd');
- select kst.event,kst.sid,kst.pid,kst.function,kst.data from x$trace kst where pid=36 and sid=17;
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10813 17 36 ktubnd ktubnd: Bind usn 3 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 3.3.1150
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-00030003-0000047e mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 69dd140000000000 0200000000000000
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 0de3140000000000 0810c76a177f0000
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=173 seq_num=180 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=173 seq_num=180 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=4, exc=4, tot=4
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=174 seq_num=181 snap_id=1
- SQL> select * from v$lock where type in('TM','TX');
- 10704 17 36 ksqrcli ksqrcl: release TX-00030003-0000047e mode=X
- 10813 17 36 ktudnx ktudnx: dec cnt xid:3.3.1150 nax:0 nbx:0
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=SX
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563c-00000000 mode=SX
- 10021 17 36 kcrf_commit_force 2ee3140000000000 2fe3140000000000
- 10005 17 36 kslwtbctx KSL WAIT BEG [log file sync] 7416/0x1cf8 1368878/0x14e32e 0/0x0 wait_id=175 seq_num=182 snap_id=1
- 10005 17 36 ksliwat KSL FACILITY WAIT fac#=3 time_waited_csecs=1
- 10005 17 36 ksliwat KSL POST RCVD poster=11 num=76 loc='ksl2.h LINE:2374 ID:kslpsr' id1=138 id2=0 name=EV type=0 fac#=3 posted=0x3 may_be_posted=1
- 10005 17 36 kslwtectx KSL WAIT END [log file sync] 7416/0x1cf8 1368878/0x14e32e 0/0x0 wait_id=175 seq_num=182 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=11126, exc=11126, tot=11126
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=176 seq_num=183 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=176 seq_num=183 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=3, exc=3, tot=3
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=177 seq_num=184 snap_id=1
insert子表:
- insert into child values(2,'sadsada');
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10813 17 36 ktubnd ktubnd: Bind usn 7 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 7.17.835
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-00070011-00000343 mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=186 seq_num=193 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=186 seq_num=193 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=3, exc=3, tot=3
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=187 seq_num=194 snap_id=1
可見子表上的插入也會獲取父表和子表mode為3的TM鎖
update父表:
- update prim set a=1 where a=1;
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=S flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=S
- 10704 17 36 ksqrcli ksqrcl: SUCCESS
- 10811 17 36 ktbgcl1 2c01000100000000 0000000000000000 0fe7140000000000 0200000000000000
- 10811 17 36 ktbgcl1 2c01000100000000 0000000000000000 27e9140000000000 b80fb86a177f0000
- 10813 17 36 ktubnd ktubnd: Bind usn 4 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 4.10.851
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-0004000a-00000353 mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=204 seq_num=211 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=204 seq_num=211 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=5, exc=5, tot=5
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=205 seq_num=212 snap_id=1
delete父表:
- delete from prim where a=4;
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=S flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=S
- 10704 17 36 ksqrcli ksqrcl: SUCCESS
- 10813 17 36 ktubnd ktubnd: Bind usn 2 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 2.0.1117
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-00020000-0000045d mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=S flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=S
- 10704 17 36 ksqrcli ksqrcl: SUCCESS
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 bae9140000000000
- 10812 17 36 ktrgcm 0200000000000000 0000000000000000 5d04000000000000
- 10812 17 36 ktrgcm 9b12c00000000000 4401000000000000 0200000000000000
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 bae9140000000000
- 10812 17 36 ktrgcm 0200000000000000 0000000000000000 5d04000000000000
- 10812 17 36 ktrgcm 9b12c00000000000 4401000000000000 0200000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 bae9140000000000
- 10812 17 36 ktrgcm 0200000000000000 0000000000000000 5d04000000000000
- 10812 17 36 ktrgcm 9b12c00000000000 4401000000000000 0200000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 bae9140000000000
- 10812 17 36 ktrgcm 0200000000000000 0000000000000000 5d04000000000000
- 10812 17 36 ktrgcm 9b12c00000000000 4401000000000000 0200000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 bae9140000000000
- 10812 17 36 ktrgcm 0200000000000000 0000000000000000 5d04000000000000
- 10812 17 36 ktrgcm 9b12c00000000000 4401000000000000 0200000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 0000000000000000
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=227 seq_num=234 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=227 seq_num=234 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=3, exc=3, tot=3
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=228 seq_num=235 snap_id=1
- SQL> delete from prim where a=4 or a=5;
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=S flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=S
- 10704 17 36 ksqrcli ksqrcl: SUCCESS
- 10813 17 36 ktubnd ktubnd: Bind usn 6 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 6.15.1290
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-0006000f-0000050a mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=S flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=S
- 10704 17 36 ksqrcli ksqrcl: SUCCESS
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 5eea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2200000000000000
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 5eea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2200000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 5eea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2200000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 5eea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2200000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 5eea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2200000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3301000100000000 0000000000000000 5eea140000000000
- 10812 17 36 ktrgcm 0000000000000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 0000000000000000 0000000000000000 0000000000000000
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 5eea140000000000 0100000000000000
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 5eea140000000000 e00fc76a177f0000
- 10812 17 36 kturCRBackoutOneChg 0100000000000000 bc06c00000000000 0e01000000000000 2200000000000000
- 10812 17 36 ktrgcm 3301000100000000 0100000000000000 0100000000000000
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=S flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqrcli ksqrcl: release TM-0001563e-00000000 mode=S
- 10704 17 36 ksqrcli ksqrcl: SUCCESS
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 5fea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2400000000000000
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 5fea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2400000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 5fea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2400000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 5fea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2400000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 5fea140000000000
- 10812 17 36 ktrgcm 0600000000000000 0f00000000000000 0a05000000000000
- 10812 17 36 ktrgcm bc06c00000000000 0e01000000000000 2400000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 0000000000000000
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=236 seq_num=243 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=236 seq_num=243 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=4, exc=4, tot=4
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=237 seq_num=244 snap_id=1
看一下此時session獲取的鎖,記住這次結果,後面會有對比。
-
SQL> select * from v$lock where type in('TM','TX');
-
-
下面來模擬一下等待,
-
sid:31
-
SQL> insert into child values(2,'12312');
-
-
1 row created.
-
-
SQL> select distinct sid from v$mystat;
-
-
SID
-
----------
-
31
-
-
sid:1169
-
-
SQL> update prim set a=1 where a=1;--hang住了
-
-
SQL> select * from v$lock where type in('TM','TX');
-
-
-
SQL> --鎖源頭查詢,帶物件和sql以及event
-
SQL> WITH sessions AS
-
2 (SELECT /*+materialize*/
-
3 sid,
-
4 blocking_session,
-
5 blocking_instance,
-
6 row_wait_obj#,
-
7 sql_id,
-
8 inst_id,
-
9 event
-
10 FROM gv$session)
-
11 SELECT LPAD(' ', 4 * (level - 1)) || s.inst_id || '.' || sid sid,
-
12 object_name,
-
13 substr(sql_text, 1, 40) sql_text,
-
14 event
-
15 FROM sessions s
-
16 LEFT OUTER JOIN dba_objects d
-
17 ON (object_id = row_wait_obj#)
-
18 LEFT OUTER JOIN gv$sql q
-
19 ON (s.sql_id = q.SQL_ID and s.inst_id = q.INST_ID)
-
20 WHERE sid IN (SELECT blocking_session FROM sessions)
-
21 OR blocking_session IS NOT NULL
-
22 CONNECT BY PRIOR sid = blocking_session
- 23 START WITH blocking_session IS NULL;
再開第三個session
-
sid:1167
-
SQL> insert into prim values(7,'dasd');--hang住了
-
- SQL> select * from v$lock where type in('TM','TX');
檢視等待鏈
二、無索引,有cascade
-
SQL> alter table child drop constraint FK_CHILD_CA;
-
-
Table altered.
-
-
SQL> alter table child add constraint FK_CHILD_CA foreign key (ca) references prim(a) on delete cascade;
-
- Table altered.
-
SQL> delete from prim where a=2 or a=4;
-
- 2 rows deleted.
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=SSX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqcnv ksqcnv: convert TM-0001563e-00000000 from=SSX to=SX flags=
- 10704 17 36 ksqcnv ksqcnv: SUCCESS
- 10813 17 36 ktubnd ktubnd: Bind usn 4 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 4.20.852
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-00040014-00000354 mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqcnv ksqcnv: convert TM-0001563e-00000000 from=SX to=SSX flags=
- 10704 17 36 ksqcnv ksqcnv: SUCCESS
- 10704 17 36 ksqcnv ksqcnv: convert TM-0001563e-00000000 from=SSX to=SX flags=
- 10704 17 36 ksqcnv ksqcnv: SUCCESS
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 bef2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0a00000000000000
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 bef2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0a00000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 bef2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0a00000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 bef2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0a00000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 bef2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0a00000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3301000100000000 0000000000000000 bef2140000000000
- 10812 17 36 ktrgcm 0000000000000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 0000000000000000 0000000000000000 0000000000000000
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 bef2140000000000 0100000000000000
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 bff2140000000000 e00fc76a177f0000
- 10812 17 36 kturCRBackoutOneChg 0100000000000000 ef00c00000000000 2801000000000000 0a00000000000000
- 10812 17 36 ktrgcm 3301000100000000 0100000000000000 0100000000000000
- 10704 17 36 ksqcnv ksqcnv: convert TM-0001563e-00000000 from=SX to=SSX flags=
- 10704 17 36 ksqcnv ksqcnv: SUCCESS
- 10704 17 36 ksqcnv ksqcnv: convert TM-0001563e-00000000 from=SSX to=SX flags=
- 10704 17 36 ksqcnv ksqcnv: SUCCESS
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 c0f2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0e00000000000000
- 10812 17 36 ktrgcm 3b01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 c0f2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0e00000000000000
- 10812 17 36 ktrgcm 3c01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 c0f2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0e00000000000000
- 10812 17 36 ktrgcm 3d01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 c0f2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0e00000000000000
- 10812 17 36 ktrgcm 3e01000100000000 0000000000000000 0000000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 c0f2140000000000
- 10812 17 36 ktrgcm 0400000000000000 1400000000000000 5403000000000000
- 10812 17 36 ktrgcm ef00c00000000000 2801000000000000 0e00000000000000
- 10812 17 36 ktrgcm 3f01000100000000 0000000000000000 0000000000000000
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=425 seq_num=432 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=425 seq_num=432 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=4, exc=4, tot=4
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=426 seq_num=433 snap_id=1
- SQL> select * from v$lock where type in('TM','TX');
是不是和沒有cascade的時候不同了,這次最終會持有子表上的mode為3的鎖,我們再深入的思考一點,SSX鎖和SX鎖是不相容的,這樣是否就意味著後進行的delete會被先進行的delete阻塞(不同session),好,現在就來模擬一下:
-
sid:1169
-
-
SQL> delete from prim where a=1;
-
-
0 rows deleted.
-
- SQL> select * from v$lock where type in('TM','TX');
-
sid:1167
-
SQL> delete from prim where a=2;--session hang住了
查詢此刻鎖的持有情況
- SQL> select * from v$lock where type in('TM','TX');
可見1167在請求mode為5的鎖,且已被阻塞
檢視等待鏈
因為delete完畢會持有子表上的SX鎖,而SX鎖與S鎖不相容,所以delete父表的session也會阻塞update父表的session,因為update回去請求子表的S鎖,而此時子表上有SX鎖,類似於子表上有事務在進行,這裡就不在論述了,徒佔篇幅。
三、有索引,無cascade
我看到有資料說,如果有索引時,對父表的操作,會級聯加一個TM RS鎖(level 2)到子表上。但我在試驗中並未看到,也許是版本差異,我也未去求證,有索引時insert與無索引時在獲取鎖方面沒有區別,這裡僅列出update和delete
建立索引:
-
SQL> alter table child drop constraint FK_CHILD_CA;
-
-
Table altered.
-
-
SQL> alter table child add constraint FK_CHILD_CA foreign key (ca) references prim(a);
-
-
Table altered.
-
-
SQL> create index ind_child_ca on child(ca);
-
- Index created.
update父表:
-
SQL> update prim set a=6 where a=6;
-
- 1 row updated.
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10811 17 36 ktbgcl1 2b01000100000000 0000000000000000 fadc140000000000 0200000000000000
- 10811 17 36 ktbgcl1 2b01000100000000 0000000000000000 cdf9140000000000 10dfb76a177f0000
- 10813 17 36 ktubnd ktubnd: Bind usn 3 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 3.28.1117
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-0003001c-0000045d mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=471 seq_num=478 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=471 seq_num=478 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=3, exc=3, tot=3
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=472 seq_num=479 snap_id=1
delete父表:
-
SQL> delete from prim where a=9;
-
- 1 row deleted.
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563c-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10704 17 36 ksqgtlctx ksqgtl: acquire TM-0001563e-00000000 mode=SX flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10811 17 36 ktbgcl1 2c01000100000000 0000000000000000 66f1140000000000 0200000000000000
- 10811 17 36 ktbgcl1 2c01000100000000 0000000000000000 58fa140000000000 78efb76a177f0000
- 10813 17 36 ktubnd ktubnd: Bind usn 4 nax 1 nbx 0 lng 0 par 0
- 10813 17 36 ktubnd ktubnd: Txn Bound xid: 4.18.854
- 10704 17 36 ksqgtlctx ksqgtl: acquire TX-00040012-00000356 mode=X flags=GLOBAL|XACT why="contention"
- 10704 17 36 ksqgtlctx ksqgtl: SUCCESS
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 c0f2140000000000 0200000000000000
- 10811 17 36 ktbgcl1 3301000100000000 0000000000000000 58fa140000000000 90eec66a177f0000
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=476 seq_num=483 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END [SQL*Net message to client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=476 seq_num=483 snap_id=1
- 10005 17 36 kslwtectx KSL WAIT END wait times (usecs) - snap=4, exc=4, tot=4
- 10005 17 36 kslwtbctx KSL WAIT BEG [SQL*Net message from client] 1650815232/0x62657100 1/0x1 0/0x0 wait_id=477 seq_num=484 snap_id=1
- SQL> select * from v$lock where type in('TM','TX');
四、有索引,有cascade
表現與無cascade時相同
五、結論
- 外來鍵無索引鎖無cascade時,update/delete父表,會在語句級別級聯一個mode為4的S鎖到子表,其中delete多少行就會級聯多少次
-
外來鍵無索引有cascade時,update父表仍會在語句級別級聯mode為4的S鎖到子表,delete時會先獲取mode為5的SSX鎖,在將其轉換成mode為3的SX鎖,而且刪除多少行就會涉及到多少次轉換
-
外來鍵有索引無cascade時,update/delete不會在語句級級聯鎖到子表,最終會持有父表和子表上的mode為3的SX鎖(無索引時只有有cascade的delete時最終會持有子表上的SX鎖)
- 外來鍵有索引有cascade時,與無cascade表現相同
參考文獻:
http://www.askmaclean.com/archives/know-more-about-oracle-row-lock.html
http://www.askmaclean.com/archives/enq-tm-contention.html
http://www.askmaclean.com/archives/know-oracle-lock-mode.html
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/26838672/viewspace-2090237/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- enq: TM - contention解決之道——外來鍵無索引導致鎖爭用ENQ索引
- Oracle等待事件之enq: TM – contentionOracle事件ENQ
- oracle 11.2.0.4 rac叢集等待事件enq: TM - contentionOracle事件ENQ
- ORACLE 如何診斷高水位爭用(enq: HW – contention)OracleENQ
- enq: TX - row lock contentionENQ
- 10.30 索引,外來鍵索引
- 外來鍵欄位未建索引引發的死鎖索引
- oracle等待事件之enq: CF – contentionOracle事件ENQ
- 等待事件enq: TX - row lock contention事件ENQ
- [20220518]enq FU - contention等待事件.txtENQ事件
- eclipse: workspace出錯導致無法啟用的解決Eclipse
- impdp導致主鍵索引的變化索引
- MySQL的index merge(索引合併)導致資料庫死鎖分析與解決方案MySqlIndex索引資料庫
- 故障排除 | enq:TX - index contention等待事件ENQIndex事件
- 奇異的enq: TX - row lock contentionENQ
- enq: TX - index contention基礎理論ENQIndex
- 效能問題,AWR High Event enq: US - contentionENQ
- [20181030]模擬分散式事務掛起導致TX鎖爭用.txt分散式
- Oracle Enqueues Wait Events 三 enq: TX - row lock contentionOracleENQAI
- enq: TX - index contention故障修復一例ENQIndex
- openGauss-外來鍵鎖增強
- 一條主鍵索引SQL導致的CPU被打滿索引SQL
- 無索引的外來鍵之主表子表DML操作實驗及結論索引
- 故障處理】佇列等待之enq: US - contention案例佇列ENQ
- 【故障處理】佇列等待之enq: US - contention案例佇列ENQ
- Mysql truncate table時解決外來鍵關聯MySql
- go 協程操作map導致的資料競爭及解決方法Go
- [20200120]oracle wait event "enq: SQ – contention" and DBA_DB_LINK_SOURCES.txtOracleAIENQ
- Redis分散式鎖(二):鎖超時後導致多個執行緒獲得鎖的解決方案Redis分散式執行緒
- 外來鍵沒有索引哪些DML操作會被阻塞索引
- h5鍵盤彈出收回導致位置錯亂解決方案H5
- JVM 輸出 GC 日誌導致 JVM 卡住,我 TM 人傻了JVMGC
- 子游標過多導致大量mutex爭用故障分析Mutex
- @Transactional開啟事務導致AbstractRoutingDataSource動態資料來源無法切換的解決方案
- 建立外來鍵時報 Cannot add foreign key constraint 解決方法AI
- 索引壞掉導致ORA-07445索引
- Win10系統下DirectDraw導致閃屏怎麼解決Win10
- 記一次升級Gradle外掛導致相容問題的解決方案Gradle
- php對外發包解決方案(導致w3wp.exe很佔cpu)PHP