问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501

什么是死锁(deadlock)?

发布网友 发布时间:2022-04-29 03:32

我来回答

4个回答

懂视网 时间:2022-05-03 14:56

【锁】Oracle死锁(DeadLock)的分类及其模拟

1  BLOG文档结构图

 

2  前言部分

2.1  导读和注意事项

各位技术爱好者,看完本文后,你可以掌握如下的技能,也可以学到一些其它你所不知道的知识,~O(∩_∩)O~:

① 死锁的概念及其trace文件

② 死锁的分类

③ 行级死锁的模拟

④ ITL的概念、ITL结构

⑤ ITL引发的死锁处理

⑥ ITL死锁的模拟

 

Tips:

① 本文在itpub(http://blog.itpub.net/26736162)、博客园(http://www.cnblogs.com/lhrbest)和微信公众号(xiaomaimiaolhr)上有同步更新。

② 文章中用到的所有代码、相关软件、相关资料及本文的pdf版本都请前往小麦苗的云盘下载,小麦苗的云盘地址见:http://blog.itpub.net/26736162/viewspace-1624453/。

③ 若网页文章代码格式有错乱,请尝试以下办法:①去博客园地址阅读,②下载pdf格式的文档来阅读。

④ 在本篇BLOG中,代码输出部分一般放在一行一列的表格中。其中,需要特别关注的地方我都用灰色背景和粉红色字体来表示,比如在下边的例子中,thread 1的最大归档日志号为33,thread 2的最大归档日志号为43是需要特别关注的地方;而命令一般使用黄色背景和红色字体标注;对代码或代码输出部分的注释一般采用蓝色字体表示。

  List of Archived Logs in backup set 11

  Thrd Seq     Low SCN    Low Time            Next SCN   Next Time

  ---- ------- ---------- ------------------- ---------- ---------

  1    32      1621589    2015-05-29 11:09:52 1625242    2015-05-29 11:15:48

  1    33      1625242    2015-05-29 11:15:48 1625293    2015-05-29 11:15:58

  2    42      1613951    2015-05-29 10:41:18 1625245    2015-05-29 11:15:49

  2    43      1625245    2015-05-29 11:15:49 1625253    2015-05-29 11:15:53

[ZHLHRDB1:root]:/>lsvg -o

T_XLHRD_APP1_vg

rootvg

[ZHLHRDB1:root]:/>

00:27:22 SQL> alter tablespace idxtbs read write;

====》2097152*512/1024/1024/1024=1G

本文如有错误或不完善的地方请大家多多指正,ITPUB留言或QQ皆可,您的批评指正是我写作的最大动力。

2.2  本文简介

写了近大半年的书了,碰到了各种困难,不过幸运的是基本上都一一克服了。前段时间工作上碰到了一个很奇怪的死锁问题,由业务发出来的SQL来看是不太可能产生死锁的,不过的的确确实实在在的产生了,那作者是碰到了哪一类的死锁呢?ITL死锁!!有关当时的案例可以参考:http://blog.itpub.net/26736162/viewspace-2124771/和http://blog.itpub.net/26736162/viewspace-2124735/。于是,作者就把死锁可能出现的情况都分类总结了一下,分享给大家,欢迎大家指出错误。本文内容也将写入作者的新书中,欢迎大家提前订阅。

第二章 死锁(DeadLock)的分类及其模拟

1  死锁简介

1.1  什么是死锁?

所谓死锁,是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程称为死锁进程。Oracle对于“死锁”是要做处理的,而不是不闻不问。

[ZFLHRDB1:oracle]:/oracle>oerr ora 60

00060, 00000, "deadlock detected while waiting for resource"

// *Cause:  Transactions deadlocked one another while waiting for resources.

// *Action: Look at the trace file to see the transactions and resources

//          involved. Retry if necessary.

 

Cause: Your session and another session are waiting for are source locked by the other. This condition is known AS a deadlock. To resolve the deadlock, one or more statements were rolled back for the other session to continue work.

Action Either: l. Enter arollback statement and re—execute all statements since the last commit or 2. Wait until the lock is released, possibly a few minutes, and then re—execute the rolled back statements.

1.2  死锁的trace文件

Oracle中产生死锁的时候会在alert告警日志文件中记录死锁的相关信息,无论单机还是RAC环境都有Deadlock这个关键词,而且当发生死锁时都会生成一个trace文件,这个文件名在alert文件中都有记载。由于在RAC环境中,是由LMD(Lock Manager Daemon)进程统一管理各个节点之间的锁资源的,所以,RAC环境中trace文件是由LMD进程来生成的。

在RAC环境中,告警日志的形式如下所示:

Mon Jun 20 10:10:56 2016

Global Enqueue Services Deadlock detected. More info in file

/u01/app/oracle/diag/rdbms/raclhr/raclhr2/trace/raclhr2_lmd0_19923170.trc.

在单机环境中,告警日志的形式如下所示:

Mon Jun 20 12:10:56 2016

ORA-00060: Deadlock detected. More info in file /u01/app/oracle/diag/rdbms/dlhr/dlhr/trace/dlhr_ora_16973880.trc.

通常来讲,对于单机环境,当有死锁发生后,在trace文件中会看到如下的日志信息:

 

图2-1 单机环境下的死锁

当看到trace文件时,需要确认一下产生锁的类型,是两行还是一行,是TX还是TM,如果只有一行那么说明是同一个SESSION,可能是自治事务引起的死锁。

对于RAC环境,当有死锁发生后,在trace文件中会看到如下的日志信息:

 

图2-2 RAC环境下的死锁

1.3  死锁的检测时间

死锁的监测时间是由隐含参数_lm_dd_interval来控制的,在Oracle 11g中,隐含参数_lm_dd_interval的值默认为10,而在Oracle 10g中该参数默认为60,单位为秒。

SYS@oraLHRDB2> SELECT A.INDX,

  2         A.KSPPINM NAME,

  3         A.KSPPDESC,

  4         B.KSPPSTVL

  5  FROM   X$KSPPI  A,

  6         X$KSPPCV B

  7  WHERE  A.INDX = B.INDX

  8  AND LOWER(A.KSPPINM) LIKE  LOWER(‘%&PARAMETER%‘);

ENTER VALUE FOR PARAMETER: _lm_dd_interval

OLD   8: AND LOWER(A.KSPPINM) LIKE  LOWER(‘%&PARAMETER%‘)

NEW   8: AND LOWER(A.KSPPINM) LIKE  LOWER(‘%_LM_DD_INTERVAL%‘)

 

      INDX NAME               KSPPDESC                       KSPPSTVL

---------- ------------------ ------------------------------ --------------------

       578 _lm_dd_interval    dd time interval in seconds    10

可以看到该隐含参数的值为10。

2  死锁的分类

有人的地方就有江湖,有资源阻塞的地方就可能有死锁。Oralce中最常见的死锁分为:行级死锁(Row-Level Deadlock)和块级死锁(Block-Level Deadlock),其中,行级死锁分为①主键、唯一索引的死锁(会话交叉插入相同的主键值),②外键未加索引,③表上的位图索引遭到并发更新,④常见事务引发的死锁(例如,两个表之间不同顺序相互更新操作引起的死锁;同一张表删除和更新之间引起的死锁),⑤自治事务引发的死锁。块级死锁主要指的是ITL(Interested Transaction List)死锁。

死锁分类图如下所示:

 

图2-3 死锁的分类图

 

2.1  行级死锁

行级锁的发生如下图所示,在A时间,TRANSACRION1和TRANSCTION2分别锁住了它们要UPDATE的一行数据,没有任何问题。但每个TRANSACTION都没有终止。接下来在B时间,它们又试图UPDATE当前正被对方TRANSACTION锁住的行,因此双方都无法获得资源,此时就出现了死锁。之所以称之为死锁,是因为无论每个TRANSACTION等待多久,这种锁都不会被释放。

 

行级锁的死锁一般是由于应用逻辑设计的问题造成的,其解决方法是通过分析trace文件定位出造成死锁的SQL语句、被互相锁住资源的对象及其记录等信息,提供给应用开发人员进行分析,并修改特定或一系列表的更新(UPDATE)顺序。

以下模拟各种行级死锁的产生过程,版本都是11.2.0.4。

2.1.1  主键、唯一索引的死锁(会话交叉插入相同的主键值)

主键的死锁其本质是唯一索引引起的死锁,这个很容易模拟出来的,新建一张表,设置主键(或创建唯一索引)后插入一个值,然后不要COMMIT,另一个会话插入另一个值,也不要COMMIT,然后再把这两个插入的值互相交换一下,在两个会话中分别插入,死锁就会产生。

会话1,sid为156:

SYS@lhrdb> set sqlprompt "_user‘@‘_connect_identifier S1> "

SYS@lhrdb S1> DROP TABLE T_DEADLOCK_PRIMARY_LHR;

 

Table dropped.

 

====>>>>> CREATE UNIQUE INDEX IDX_TEST_LHR ON T_DEADLOCK_PRIMARY_LHR(ID);

SYS@lhrdb S1> CREATE TABLE T_DEADLOCK_PRIMARY_LHR(ID NUMBER PRIMARY KEY);

 

Table created.

 

SYS@lhrdb S1> select userenv(‘sid‘) from dual;

 

USERENV(‘SID‘)

--------------

           156

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(1);

 

1 row created.

 

SYS@lhrdb S1>

会话2,sid为156:

SYS@lhrdb> set sqlprompt "_user‘@‘_connect_identifier S2> "

SYS@lhrdb S2> select userenv(‘sid‘) from dual;

 

USERENV(‘SID‘)

--------------

           191

 

SYS@lhrdb S2> INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(2);

 

1 row created.

 

SYS@lhrdb S2> INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(1);

 

 

 

====>>>>> 产生了阻塞

 

 

SELECT NVL(A.SQL_ID, A.PREV_SQL_ID) SQL_ID,

       A.BLOCKING_SESSION,

       A.SID,

       A.SERIAL#,

       A.LOGON_TIME,

       A.EVENT

  FROM GV$SESSION A

 WHERE A.SID IN (156,191)

 ORDER BY A.LOGON_TIME;

 

156阻塞了191会话,即会话1阻塞了会话2。

会话1再次插入数据:

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(2);

 

 

 

 

====>>>>> 产生了阻塞

 

此时,去会话2看的时候,已经报出了死锁的错误:

 

此时的阻塞已经发生了变化:

 

告警日志:

Fri Sep 23 09:03:11 2016

ORA-00060: Deadlock detected. More info in file /oracle/app/oracle/diag/rdbms/lhrdb/lhrdb/trace/lhrdb_ora_25886852.trc.

其内容可以看到很经典的一段:

Deadlock graph:

                       ---------Blocker(s)--------  ---------Waiter(s)---------

Resource Name          process session holds waits  process session holds waits

TX-0008000c-000008dc        38     191     X             29     156           S

TX-00030016-00000892        29     156     X             38     191           S

 

session 191: DID 0001-0026-00000115     session 156: DID 0001-001D-0000000D

session 156: DID 0001-001D-0000000D     session 191: DID 0001-0026-00000115

 

Rows waited on:

  Session 191: no row

  Session 156: no row

 

这就是主键的死锁,模拟完毕。

此时,若是会话2执行提交后,会话1就会报错,违反唯一约束:

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(2);

 

 

INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(2)

*

ERROR at line 1:

ORA-00001: unique constraint (SYS.SYS_C0011517) violated

一、 脚本

会话1:

DROP TABLE T_DEADLOCK_PRIMARY_LHR;

CREATE TABLE T_DEADLOCK_PRIMARY_LHR(ID NUMBER PRIMARY KEY);

--CREATE UNIQUE INDEX IDX_TEST_LHR ON T_DEADLOCK_PRIMARY_LHR(ID);

select userenv(‘sid‘) from dual;

INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(1);

 

会话2:

select userenv(‘sid‘) from dual;

INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(2);

INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(1);

 

 

会话1:---死锁产生

INSERT INTO T_DEADLOCK_PRIMARY_LHR VALUES(2);

 

 

SELECT NVL(A.SQL_ID, A.PREV_SQL_ID) SQL_ID,

       A.BLOCKING_SESSION,

       A.SID,

       A.SERIAL#,

       A.LOGON_TIME,

       A.EVENT

  FROM GV$SESSION A

WHERE A.SID IN (156,191)

ORDER BY A.LOGON_TIME;

 

2.1.2  外键的死锁(外键未加索引)

外键未加索引很容易导致死锁。在以下两种情况下,Oracle在修改父表后会对子表加一个全表锁:

1. 如果更新了父表的主键,由于外键上没有索引,所以子表会被锁住。

2. 如果删除了父表中的一行,由于外键上没有索引,整个子表也会被锁住。

总之,就是更新或者删除父表的主键,都会导致对其子表加一个全表锁。

如果父表存在删除记录或者更改外键列的情形,那么就需要在子表上为外键列创建索引。

除了全表锁外,在以下情况下,未加索引的外键也可能带来问题:

1. 如果有ON DELETE CASCADE,而且没有对子表加索引:例如,EMP是DEPT的子表,DELETE DEPTNO = 10应该CASCADE(级联)至EMP[4]。如果EMP中的DEPTNO没有索引,那么删除DEPT表中的每一行时都会对EMP做一个全表扫描。这个全表扫描可能是不必要的,而且如果从父表删除多行,父表中每删除一行就要扫描一次子表。

2. 从父表查询子表:再次考虑EMP/DEPT例子。利用DEPTNO查询EMP表是相当常见的。如果频繁地运行以下查询(例如,生成一个报告),你会发现没有索引会使查询速度变慢: 

SELECT * FROM DEPT, EMP 

WHERE EMP.DEPTNO = DEPT.DEPTNO

AND DEPT.DEPTNO = :X; 

那么,什么时候不需要对外键加索引呢?答案是,一般来说,当满足以下条件时不需要加索引: 

     1.  没有从父表删除行。

     2.  没有更新父表的惟一键/主键值。

     3.  没有从父表联结子表。

如果满足上述全部3个条件,那你完全可以跳过索引,不需要对外键加索引。不过个人还是强烈建议对子表添加索引,既然已经创建了外键,就不在乎再多一个索引吧,因为一个索引所增加的代价比如死锁,与缺失这个索引所带来的问题相比,是微不足道的。 

子表上为外键列建立索引,可以:

1)提高针对外键列的查询或改动性能

2)减小表级锁粒度,降低死锁发生的可能性

外键的死锁可以这样通俗的理解:有两个表A和B:A是父表,B是子表。如果没有在B表中的外键加上索引,那么A表在更新或者删除主键时,都会在表B上加一个全表锁。这是为什么呢?因为我们没有给外键加索引,在更新或者删除A表主键的时候,需要查看子表B中是否有对应的记录,以判断是否可以更新删除。那如何查找呢?当然只能在子表B中一条一条找了,因为我们没有加索引吗。既然要在子表B中一条一条地找,那就得把整个子表B都锁定了。由此就会导致以上一系列问题。

实验过程:

会话1首先建立子表和父表

SYS@lhrdb S1> CREATE TABLE T_DEADLOCK_P_LHR (ID NUMBER PRIMARY KEY, NAME VARCHAR2(30)); --父表的主键

 

Table created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_P_LHR VALUES (1, ‘A‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_P_LHR VALUES (2, ‘B‘);

 

1 row created.

 

SYS@lhrdb S1> CREATE TABLE T_DEADLOCK_F_LHR (FID NUMBER, FNAME VARCHAR2(30), FOREIGN KEY (FID) REFERENCES T_DEADLOCK_P_LHR);   --子表的外键

 

Table created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_F_LHR VALUES (1, ‘C‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_F_LHR VALUES (2, ‘D‘);

 

1 row created.

 

SYS@lhrdb S1>

SYS@lhrdb S1> COMMIT;

 

Commit complete.

 

 

会话1执行一个删除操作,这时候在子表和父表上都加了一个Row-X(SX)锁

SYS@lhrdb S1> delete from T_DEADLOCK_F_LHR where FID=1;

 

1 row deleted.

 

SYS@lhrdb S1> delete from T_DEADLOCK_P_LHR where id=1;

 

1 row deleted.

 

查询会话1的锁信息:

SELECT LK.SID,

       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, 

       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(+)

 AND SE.SID IN (156,191)

 ORDER BY SID;

 

BLOCK为0表示没有阻塞其它的锁。

 

会话2:执行另一个删除操作,发现这时候第二个删除语句等待

SYS@lhrdb S2> delete from T_DEADLOCK_F_LHR where FID=2;

 

1 row deleted.

 

SYS@lhrdb S2> delete from T_DEADLOCK_P_LHR where id=2;

 

 

 

 

====>>>>> 产生了阻塞

 

 

 

BLOCK为1表示阻塞了其它的锁。

会话1执行删除语句,死锁发生

SYS@lhrdb S1> delete from T_DEADLOCK_P_LHR where id=1;

 

 

 

====>>>>> 产生了阻塞,而会话2产生了死锁

 

 

SYS@lhrdb S2> delete from T_DEADLOCK_P_LHR where id=2;

 

 

 

delete from T_DEADLOCK_P_LHR where id=2

            *

ERROR at line 1:

ORA-00060: deadlock detected while waiting for resource

 

 

 

 

 

 

告警日志:

Fri Sep 23 10:31:10 2016

ORA-00060: Deadlock detected. More info in file /oracle/app/oracle/diag/rdbms/lhrdb/lhrdb/trace/lhrdb_ora_25886852.trc.

 

查看内容:

*** 2016-09-23 10:31:10.212

DEADLOCK DETECTED ( ORA-00060 )

 

[Transaction Deadlock]

 

The following deadlock is not an ORACLE error. It is a

deadlock due to user error in the design of an application

or from issuing incorrect ad-hoc SQL. The following

information may aid in determining the deadlock:

 

Deadlock graph:

                       ---------Blocker(s)--------  ---------Waiter(s)---------

Resource Name          process session holds waits  process session holds waits

TM-00017731-00000000        38     191    SX   SSX       29     156    SX   SSX

TM-00017731-00000000        29     156    SX   SSX       38     191    SX   SSX

 

session 191: DID 0001-0026-00000115     session 156: DID 0001-001D-0000000D

session 156: DID 0001-001D-0000000D     session 191: DID 0001-0026-00000115

 

Rows waited on:

  Session 191: no row

  Session 156: no row

 

回滚会话建立外键列上的索引:

SYS@lhrdb S1> rollback;

 

Rollback complete.

 

SYS@lhrdb S1> create index ind_p_f_fid on T_DEADLOCK_F_LHR(fid);

 

Index created.

重复上面的步骤会话1删除子表记录:

---会话1:

SYS@lhrdb S1> delete from T_DEADLOCK_F_LHR where FID=1;

 

1 row deleted.

 

SYS@lhrdb S1> delete from T_DEADLOCK_P_LHR where id=1;

 

1 row deleted.

 

SYS@lhrdb S1>

 

---会话2:

SYS@lhrdb S2> delete from T_DEADLOCK_F_LHR where FID=2;

 

1 row deleted.

 

SYS@lhrdb S2> delete from T_DEADLOCK_P_LHR where id=2;

 

1 row deleted.

所有的删除操作都可以成功执行,也没有阻塞的生成,重点就是在外键列上建立索引。

 

 

一、 脚本


--主表

DROP TABLE T_DEADLOCK_P_LHR;

CREATE TABLE T_DEADLOCK_P_LHR (ID NUMBER PRIMARY KEY, NAME VARCHAR2(30)); --父表的主键

INSERT INTO T_DEADLOCK_P_LHR VALUES (1, ‘A‘);

INSERT INTO T_DEADLOCK_P_LHR VALUES (2, ‘B‘);

 

--子表

DROP TABLE T_DEADLOCK_F_LHR;

CREATE TABLE T_DEADLOCK_F_LHR (FID NUMBER, FNAME VARCHAR2(30), FOREIGN KEY (FID) REFERENCES T_DEADLOCK_P_LHR);  --子表的外键

INSERT INTO T_DEADLOCK_F_LHR VALUES (1, ‘C‘);

INSERT INTO T_DEADLOCK_F_LHR VALUES (2, ‘D‘);

 

COMMIT;

 

 

---执行一个删除操作,这时候在子表和父表上都加了一个Row-S(SX)锁

delete from T_DEADLOCK_F_LHR where FID=1;

delete from T_DEADLOCK_P_LHR where id=1;

---会话2:执行另一个删除操作,发现这时候第二个删除语句等待

delete from T_DEADLOCK_F_LHR where FID=2;

delete from T_DEADLOCK_P_LHR where id=2;

---会话1:死锁发生

delete from T_DEADLOCK_P_LHR where id=1;

 

 

---回滚会话建立外键列上的索引:

create index ind_p_f_fid on T_DEADLOCK_F_LHR(fid);

 

 

--重复上面的步骤会话1删除子表记录: 

---会话1:

delete from T_DEADLOCK_F_LHR where FID=1;

delete from T_DEADLOCK_P_LHR where id=1;

---会话2:执行另一个删除操作,发现这时候第二个删除语句等待

delete from T_DEADLOCK_F_LHR where FID=2;

delete from T_DEADLOCK_P_LHR where id=2;

 

2.1.3  位图(BITMAP)索引死锁

表上的位图索引遭到并发更新也很容易产生死锁。在有位图索引存在的表上面,其实很容易就引发阻塞与死锁。这个阻塞不是发生在表上面,而是发生在索引上。因为位图索引锁定的范围远远比普通的b-tree索引锁定的范围大。

SYS@lhrdb S1> CREATE TABLE T_DEADLOCK_BITMAP_LHR (ID NUMBER , NAME VARCHAR2(10));

 

Table created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (1,‘A‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (1,‘B‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (1,‘C‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (2,‘A‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (2,‘B‘);

 

1 row created.

 

SYS@lhrdb S1> INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (2,‘C‘);

 

1 row created.

 

SYS@lhrdb S1> COMMIT;

 

Commit complete.

 

--那么在ID列上建bitmap index的话,所有ID=1的会放到一个位图中,所有ID=2的是另外一个位图,而在执行DML操作的时候,锁定的将是整个位图中的所有行,而不仅仅是DML涉及到的行。由于锁定的粒度变粗,bitmap index更容易导致死锁的发生。

会话1:此时所有ID=1的行都被锁定

SYS@lhrdb S1> CREATE BITMAP INDEX IDX_DEADLOCK_BITMAP_LHR ON T_DEADLOCK_BITMAP_LHR(ID);

 

Index created.

 

SYS@lhrdb S1> UPDATE T_DEADLOCK_BITMAP_LHR SET ID=3 WHERE ID=1 AND NAME=‘A‘;

 

1 row updated.

 

会话2:此时所有ID=2的行都被锁定

SYS@lhrdb S2> UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘A‘;

 

1 row updated.

 

会话1:此时会话被阻塞

SYS@lhrdb S1> UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘B‘;

 

 

 

 

====>>>>> 产生了阻塞

 

 

会话2:会话被阻塞

SYS@lhrdb S2> UPDATE T_DEADLOCK_BITMAP_LHR SET ID=3 WHERE ID=1 AND NAME=‘B‘;

 

 

 

====>>>>> 产生了阻塞

再回到SESSION 1,发现系统检测到了死锁的发生

SYS@lhrdb S1> UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘B‘;

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘B‘

       *

ERROR at line 1:

ORA-00060: deadlock detected while waiting for resource

 

告警日志:

Fri Sep 23 11:20:21 2016

ORA-00060: Deadlock detected. More info in file /oracle/app/oracle/diag/rdbms/lhrdb/lhrdb/trace/lhrdb_ora_19726460.trc.

内容:

*** 2016-09-23 11:26:51.264

DEADLOCK DETECTED ( ORA-00060 )

 

[Transaction Deadlock]

 

The following deadlock is not an ORACLE error. It is a

deadlock due to user error in the design of an application

or from issuing incorrect ad-hoc SQL. The following

information may aid in determining the deadlock:

 

Deadlock graph:

                       ---------Blocker(s)--------  ---------Waiter(s)---------

Resource Name          process session holds waits  process session holds waits

TX-0009000e-00000b0f        29     156     X             38     191           S

TX-00070001-00000b2c        38     191     X             29     156           S

 

session 156: DID 0001-001D-0000000D     session 191: DID 0001-0026-00000115

session 191: DID 0001-0026-00000115     session 156: DID 0001-001D-0000000D

 

Rows waited on:

  Session 156: obj - rowid = 00017734 - AAAXc0AAAAAAAAAAAA

  (dictionary objn - 96052, file - 0, block - 0, slot - 0)

  Session 191: obj - rowid = 00017734 - AAAXc0AAAAAAAAAAAA

  (dictionary objn - 96052, file - 0, block - 0, slot - 0)

 

 

死锁发生的根本原因是对于资源的排他锁定顺序不一致。上面的试验中,session1对于bitmap index中的2个位图是先锁定ID=1的位图,然后请求ID=2的位图,而在此之前ID=2的位图已经被session2锁定。session2则先锁定ID=2的位图,然后请求ID=2的位图,而此前ID=1的位图已经被session1锁定。于是,session1等待session2释放ID=2的位图上的锁,session2等待session1释放ID=1的位图上的锁,死锁就发生了

而如果我们创建的是普通的B*Tree index,重复上面的试验则不会出现任何的阻塞和死锁,这是因为锁定的只是DML操作涉及到的行,而不是所有ID相同的行。

一、 脚本

CREATE TABLE T_DEADLOCK_BITMAP_LHR (ID NUMBER , NAME VARCHAR2(10));

INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (1,‘A‘);

INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (1,‘B‘);

INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (1,‘C‘);

INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (2,‘A‘);

INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (2,‘B‘);

INSERT INTO T_DEADLOCK_BITMAP_LHR VALUES (2,‘C‘);

COMMIT;

 

CREATE BITMAP INDEX IDX_DEADLOCK_BITMAP_LHR ON T_DEADLOCK_BITMAP_LHR(ID);

 

--会话1:此时所有ID=1的行都被锁定

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=3 WHERE ID=1 AND NAME=‘A‘;

--会话2:此时所有ID=2的行都被锁定

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘A‘;

 

--会话1:此时会话被阻塞

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘B‘;

 

--会话2:会话被阻塞,再回到SESSION 1,发现系统检测到了死锁的发生

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=3 WHERE ID=1 AND NAME=‘B‘;

 

--会话1:

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘B‘;

UPDATE T_DEADLOCK_BITMAP_LHR SET ID=4 WHERE ID=2 AND NAME=‘B‘

*

ERROR at line 1:

ORA-00060: deadlock detected while waiting for resource

 

2.1.4  常见事务引发的死锁

如果你有两个会话,每个会话都持有另一个会话想要的资源,此时就会出现死锁(deadlock)。例如,如果我的数据库中有两个表A和B,每个表中都只有一行,就可以很容易地展示什么是死锁。我要做的只是打开两个会话(例如,两个SQL*Plus会话)。在会话A中更新表A,并在会话B中更新表B。现在,如果我想在会话B中更新表A,就会阻塞。会话A已经锁定了这一行。这不是死锁;只是阻塞而已。因为会话A还有机会提交或回滚,这样会话B就能继续。如果我再回到会话A,试图更新表B,这就会导致一个死锁。要在这两个会话中选择一个作为“牺牲品”,让它的语句回滚。 

想要更新表B的会话A还阻塞着,Oracle不会回滚整个事务。只会回滚与死锁有关的某条语句。会话B仍然锁定着表B中的行,而会话A还在耐心地等待这一行可用。收到死锁消息后,会话B必须决定将表B上未执行的工作提交还是回滚,或者继续走另一条路,以后再提交。一旦这个会话执行提交或回滚,另一个阻塞的会话就会继续,好像什么也没有发生过一样。

一、 模拟一:两个表之间不同顺序相互更新操作引起的死锁

1、创建两个简单的表A和B,每个表中仅仅包含一个字段id。

[ZFZHLHRDB2:oracle]:/oracle>ORACLE_SID=raclhr2

[ZFZHLHRDB2:oracle]:/oracle>sqlplus / as sysdba

 

SQL*Plus: Release 11.2.0.4.0 Production on Mon Jun 20 09:40:24 2016

 

Copyright (c) 1982, 2013, Oracle.  All rights reserved.

 

 

Connected to:

Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production

With the Partitioning, Real Application Clusters, Automatic Storage Management, OLAP,

Data Mining and Real Application Testing options

 

SYS@raclhr2> select * from v$version;

 

BANNER

--------------------------------------------------------------------------------

Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production

PL/SQL Release 11.2.0.4.0 - Production

CORE    11.2.0.4.0      Production

TNS for IBM/AIX RISC System/6000: Version 11.2.0.4.0 - Production

NLSRTL Version 11.2.0.4.0 - Production

 

SYS@raclhr2> show parameter cluster

 

NAME                                 TYPE        VALUE

------------------------------------ ----------- ------------------------------

cluster_database                     boolean     TRUE

cluster_database_instances           integer     2

cluster_interconnects                string

 

SYS@raclhr2> create table A (id int);

 

Table created.

 

SYS@raclhr2> create table B (id int);

 

Table created.

 

2、每张表中仅初始化一条数据

SYS@raclhr2> insert into A values (1);

 

1 row created.

 

SYS@raclhr2> insert into B values (2);

 

1 row created.

 

SYS@raclhr2> commit;

 

Commit complete.

 

3、在第一个会话session1中更新表A中的记录“1”为“10000”,不提交;在第二个会话session2中更新表B中的记录“2”为“20000”,不提交

session1的情况:

SYS@raclhr2> insert into A values (1);

 

1 row created.

 

SYS@raclhr2> insert into B values (2);

 

1 row created.

 

SYS@raclhr2> commit;

 

Commit complete.

 

SYS@raclhr2> SELECT a.SID,

  2         b.SERIAL# ,

  3         c.SPID

  4  FROM   v$mystat  a,

  5         v$session b ,

  6         v$process c

  7  WHERE  a.SID = b.SID

  8  and b.PADDR=c.ADDR

  9  AND    rownum = 1;

 

       SID    SERIAL# SPID

---------- ---------- ------------------------

       133          3 20906088

 

SYS@raclhr2>  update A set id = 10000 where id = 1;

 

1 row updated.

 

session2的情况:

SYS@raclhr2> SELECT a.SID,

  2         b.SERIAL# ,

  3         c.SPID

  4  FROM   v$mystat  a,

  5         v$session b ,

  6         v$process c

  7  WHERE  a.SID = b.SID

  8  and b.PADDR=c.ADDR

  9  AND    rownum = 1;

 

       SID    SERIAL# SPID

---------- ---------- ------------------------

       195         21 11010172

 

SYS@raclhr2> update B set id = 20000 where id = 2;

 

1 row updated.

 

SYS@raclhr2>

 

4、此时,没有任何问题发生。OK,现在注意一下下面的现象,我们再回到会话session1中,更新表B的记录,此时出现了会话阻塞,更新hang住不能继续。

SYS@raclhr2> update B set id = 10000 where id = 2;

 

这里出现了“锁等待”(“阻塞”)的现象,原因很简单,因为在session2中已经对这条数据执行过update操作没有提交表示已经对该行加了行级锁。

SYS@raclhr2> set line 9999

SYS@raclhr2> SELECT A.INST_ID,

  2         A.SID,

  3         A.SERIAL#,

  4         A.SQL_ID,

  5         A.BLOCKING_INSTANCE,

  6         A.BLOCKING_SESSION,

  7         A.EVENT

  8    FROM gv$session a

  9   WHERE a.USERNAME IS NOT NULL

10     and a.STATUS = ‘ACTIVE‘

11     and a.BLOCKING_SESSION IS NOT NULL ;

 

   INST_ID        SID    SERIAL# SQL_ID        BLOCKING_INSTANCE BLOCKING_SESSION EVENT

---------- ---------- ---------- ------------- ----------------- ---------------- ------------------------------

         2        133          3 6k793mj0duubw                 2              195 enq: TX - row lock contention

我们可以通过v$session视图看到,实例2的195阻塞了实例2的133会话,即本实验中的session2阻塞了session1。

6、接下来再执行一条SQL后死锁就会产生了:在session2中,更新表A的记录

SYS@raclhr2> update A set id = 10000 where id = 1;

这里还是长时间的等待,但是这里发生了死锁,这个时候我们去第一个会话session1中看一下,原先一直在等待的SQL语句报了如下的错误:

SYS@raclhr2> update B set id = 10000 where id = 2;

update B set id = 10000 where id = 2

       *

ERROR at line 1:

ORA-00060: deadlock detected while waiting for resource

 

若此时查询v$session视图可以看到实例2的133阻塞了实例2的195会话,即本实验中的session1阻塞了session2,和刚刚的阻塞情况相反,说明oracle做了自动处理:

SYS@raclhr2> set line 9999

SYS@raclhr2> SELECT A.INST_ID,

  2         A.SID,

  3         A.SERIAL#,

  4         A.SQL_ID,

  5         A.BLOCKING_INSTANCE,

  6         A.BLOCKING_SESSION,

  7         A.EVENT

  8    FROM gv$session a

  9   WHERE a.USERNAME IS NOT NULL

10     and a.STATUS = ‘ACTIVE‘

11      and a.BLOCKING_SESSION IS NOT NULL ;

 

   INST_ID        SID    SERIAL# SQL_ID        BLOCKING_INSTANCE BLOCKING_SESSION EVENT

---------- ---------- ---------- ------------- ----------------- ---------------- ------------------------------

         2        195         21 5q7t3877fdu3n                 2              133 enq: TX - row lock contention

 

更进一步:查看一下alert警告日志文件发现有如下的记录:

Mon Jun 20 10:10:56 2016

Global Enqueue Services Deadlock detected. More info in file

/u01/app/oracle/diag/rdbms/raclhr/raclhr2/trace/raclhr2_lmd0_19923170.trc.

若是单机环境,报警日志为:

ORA-00060: Deadlock detected. More info in file /oracle/app/oracle/diag/rdbms/dlhr/dlhr/trace/dlhr_ora_16973880.trc.

 

可以看到该文件是由lmd进程生成的,为rac的特有进程,完成CacheFusion的作用,再进一步:看看系统自动生成的trace文件中记录了什么:

user session for deadlock lock 0x70001001569c378

  sid: 133 ser: 3 audsid: 4294967295 user: 0/SYS

    flags: (0x41) USR/- flags_idl: (0x1) BSY/-/-/-/-/-

    flags2: (0x40009) -/-/INC

  pid: 60 O/S info: user: oracle, term: UNKNOWN, ospid: 20906088

    image: oracle@ZFZHLHRDB2 (TNS V1-V3)

  client details:

    O/S info: user: oracle, term: pts/0, ospid: 16122014

    machine: ZFZHLHRDB2 program: sqlplus@ZFZHLHRDB2 (TNS V1-V3)

    application name: sqlplus@ZFZHLHRDB2 (TNS V1-V3), hash value=346557658

  current SQL:

  update B set id = 10000 where id = 2

DUMP LOCAL BLOCKER: initiate state dump for DEADLOCK

  possible owner[60.20906088] on resource TX-00140013-0000072D

 

 

user session for deadlock lock 0x700010015138660

  sid: 195 ser: 21 audsid: 4294967295 user: 0/SYS

    flags: (0x41) USR/- flags_idl: (0x1) BSY/-/-/-/-/-

    flags2: (0x40009) -/-/INC

  pid: 46 O/S info: user: oracle, term: UNKNOWN, ospid: 11010172

    image: oracle@ZFZHLHRDB2 (TNS V1-V3)

  client details:

    O/S info: user: oracle, term: pts/1, ospid: 16646154

    machine: ZFZHLHRDB2 program: sqlplus@ZFZHLHRDB2 (TNS V1-V3)

    application name: sqlplus@ZFZHLHRDB2 (TNS V1-V3), hash value=346557658

  current SQL:

  update A set id = 10000 where id = 1

DUMP LOCAL BLOCKER: initiate state dump for DEADLOCK

  possible owner[46.11010172] on resource TX-000B0018-00000416

 

若是单机环境比较明显:

[Transaction Deadlock]

 

The following deadlock is not an ORACLE error. It is a

deadlock due to user error in the design of an application

or from issuing incorrect ad-hoc SQL. The following

information may aid in determining the deadlock:

 

Deadlock graph:

                       ---------Blocker(s)--------  ---------Waiter(s)---------

Resource Name          process session holds waits  process session holds waits

TX-000d0005-00000047        41      37     X             25      34           X

TX-0008000a-0000036b        25      34     X             41      37           X

 

session 37: DID 0001-0029-00000003session 34: DID 0001-0019-0000000D

session 34: DID 0001-0019-0000000Dsession 37: DID 0001-0029-00000003

 

Rows waited on:

  Session 37: obj - rowid = 00015FE7 - AAAV/nAABAAAXeBAAA<

热心网友 时间:2022-05-03 12:04

所谓死锁: 是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程称为死锁进程。

由于资源占用是互斥的,当某个进程提出申请资源后,使得有关进程在无外力协助下,永远分配不到必需的资源而无法继续运行,这就产生了一种特殊现象死锁。一种情形,此时执行程序中两个或多个线程发生永久堵塞(等待),每个线程都在等待被其他线程占用并堵塞了的资源。例如,如果线程A锁住了记录1并等待记录2,而线程B锁住了记录2并等待记录1,这样两个线程就发生了死锁现象。计算机系统中,如果系统的资源分配策略不当,更常见的可能是程序员写的程序有错误等,则会导致进程因竞争资源不当而产生死锁的现象。
排除方法:
1、撤消陷于死锁的全部进程;
2、逐个撤消陷于死锁的进程,直到死锁不存在;
3、从陷于死锁的进程中逐个强迫放弃所占用的资源,直至死锁消失;
4、从另外一些进程那里强行剥夺足够数量的资源分配给死锁进程,以解除死锁状态。

热心网友 时间:2022-05-03 13:22

两个进程都在等待对方执行完毕才能继续往下执行的时候就发生了死锁。结果就是两个进程都陷入了无限的等待中。

热心网友 时间:2022-05-03 14:56

这是一个计算机方面的专业术语吧,线程同步方面的!
声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com
焦作有哪些旅行景点值得一去? 河南穿古装去的地方 AE入门从学会套模板开始,AE模板套用简易教程,看起来复杂的AE其实也很... ae怎么套用模板ae模版的使用方法 北京通州区有什么好玩的地方吗 请问现在有哪些看电影的网站?越多越好~~ 帮初中女儿请假一天讲身体不适,班主任却跟别的老师讲她得了大病,该如何... 梦见家中被盗空只剩一件绿色衣的预兆 梦见美丽沙穗 美版4s内置卡贴有什么危害? 福清福泽公馆七号楼2303x建筑面积140平方室内设计剩下多少平方_百度问一问 在福州优秀的室内设计师哪里找,有没有适合的推荐几个参考下? 福州哪些大学学校有室内设计专业 福州的优秀室内设计师,当下时兴什么样的室内设计理念? 亚美广告有限公司福清分公司怎么样? 福建诚丰装饰设计工程有限公司怎么样? 福清哪里设计软件最专业的? 主要 CAD ,3D!要专业的!! 福建中铭装饰设计工程有限公司怎么样? 福清思新广告装潢有限公司怎么样? 福清福泽公馆七号楼2303x建筑面积140平方室内设计剩下多少平方_百度问一问 福建福州和福清,有什么学校的中专或者5连专有学室内装修设计的? 福清哪里设计软件最专业的? 主要 CAD ,3D!要专业的!! 在请教下 现在的室内设计 都用些什么别的软件吗 福清市凯悦翁装饰设计工程有限公司怎么样? 福清市康家装饰设计工程有限公司怎么样? 福清市美星装饰设计有限公司怎么样? 福清市伍家装饰设计有限公司怎么样? 问个问题,wps表格与excel功能相同吗 wps和excel一样吗 wps表格和excel有什么区别 苹果8充电口坏了能卖多少钱 什么是死锁,死锁的原因是什么 什么是死锁? 什么是死锁?产生死锁的原因是什么? 用手机看视频,里面有个二维码,我很想扫描这个二维码,我怎么样才能扫描到呢? 计算机死锁是什么意思? 何谓死锁 产生死锁的原因和必要条件是什么 为什么说死锁 什么是死锁?解决死锁有那几种策略?这些策略分别有哪些实现方法?_百度... 什么是死锁及死锁的必要条件和解决方法 死锁的概念是什么? PPT图标链接`点这个图标能在下面出现文字`请问怎么设置超链接 什么是死锁,简述死锁发生的四个必要条件,如何避免死锁 什么是死锁?如何解决死锁问题 如何让win10不记录打开的文件 姓罗的男孩取什么名字好 2015年2月3日生罗姓男孩求取个好名 求介绍一本好用的中考作文参考书 中学生作文学习辅导书推荐 中考作文辅导书 曹操送宁波有没有分公司 手机拍的照片在手机上看非常清晰,怎么放到电脑上看就不清晰了呢