今天生成了生产库前几日的AWR报告,发现等待事件中出现了一个陌生的event--enq: HW - contention,google一下是ASSM(Auto Segment Space Management)的表空间上扩展时引起的等待。通常是有LOB大对象的表的并发插入、更新引起的LOB Segment High Water Mark并发争用;了解了之后开始解决问题。

  原因:为防止多个进程同时修改HWM而提供的锁称为HW锁。想要移动HWM的进程必须获得HW锁。若在获取HW锁过程中发生争用,则等待enq: HW - contention事件。HW锁争用大部分是大量执行insert所引发的。

1.从语句跟踪
1)首先查出是哪个语句引起的争用

select SQL_ID,EVENT,to_char(sample_time,'yyyy/mm/dd hh24:mi:ss') as dt from v$active_session_history
where event like 'enq: HW%'
order by sample_time desc;

SQL_ID EVENT DT
------------- ---------------------------------------------------------------- ---------------------------------------------------------------------------
ctad8v29f07wm enq: HW - contention 2013/10/18 16:33:00
ctad8v29f07wm enq: HW - contention 2013/10/18 16:33:00
ctad8v29f07wm enq: HW - contention 2013/10/18 16:33:00

.......

2)根据sql_id确定是哪个表的什么字段争用引起等待
SELECT * FROM V$SQLTEXT WHERE SQL_ID='ctad8v29f07wm';

ADDRESS HASH_VALUE SQL_ID COMMAND_TYPE PIECE SQL_TEXT
---------------- ---------- ------------- ------------ ---------- ----------------------------------------------------------------
000000045F9FE0D0 2464161683 ctad8v29f07wm 6 1 :p2))
000000045F9FE0D0 2464161683 ctad8v29f07wm 6 0 UPDATE "SYS_ENTRYDIC" SET "APPSETS" = :p1 WHERE (("BHAPPITEM" =

--至此明白是SYS_ENTRYDIC上的APPSETS字段批量更新或插入引起的等待,结合应用业务也确实如此,此字段更新,插入频率较高。

2.直接定位文件号和块号,得出表名字段名
1)查出哪个文件哪个块
select event,p1,p2,p3 from v$session_wait where event like '%contention%';

EVENT P1 P2 P3
---------------------------------------------------------------- ---------- ---------- ----------
enq: HW - contention 1213661190 6 20974058
enq: HW - contention 1213661190 6 20974058
enq: HW - contention 1213661190 6 20974058
enq: HW - contention 1213661190 6 20974058
enq: HW - contention 1213661190 6 20974058
enq: HW - contention 1213661190 6 20974058
enq: HW - contention 1213661190 6 20974058

select dbms_utility.data_block_address_block(20974058) as blk#,dbms_utility.data_block_address_file(20974058) as file# from dual;

BLK# FILE#
---------- ----------
2538 5

2)得到段名
select owner,segment_name,segment_type, segment_name,block_id,blocks
from dba_extents
where file_id = 5
and 2538 between block_id and block_id+blocks-1;

OWNER SEGMENT_NAME SEGMENT_TYPE SEGMENT_NAME BLOCK_ID BLOCKS
------------------------------ -------------------------------------------------- ------------------ -------------------------------------------------- ---------- ----------
BOLAN SYS_LOB0000075975C00004$$ LOBSEGMENT SYS_LOB0000075975C00004$$ 2536 8

3)确认表名字段名
select table_name,column_name,segment_name,index_name from dba_lobs
where segment_name='SYS_LOB0000075975C00004$$';

TABLE_NAME COLUMN_NAME SEGMENT_NAME INDEX_NAME
------------------------------ -------------------------------------------------- -------------------------------------------------- ------------------------------
SYS_ENTRYDIC APPSETS SYS_LOB0000075975C00004$$ SYS_IL0000075975C00004$$

1和2中相互验证了引起争用的段,解决方式分两种,在ASSM表空间之内的:

a) As temporary workaround, manually add extra space to the LOB segment

ALTER TABLE <lob_table> MODIFY LOB (<column_name>) (allocate extent (size <extent size>));
或者
b) It may related Bug 6376915.
Refer to Note 6376915.8 “Bug 6376915 HW enqueue contention for ASSM LOB segments”
In 10.2.0.4 or above, this fix has been included, and can be enabled by setting event 44951 to a value
between 1 and 1024. A higher value would be more beneficial in reducing contention.
EVENT=”44951 TRACE NAME CONTEXT FOREVER, LEVEL < 1 – 1024 >”
语句如下:
alter system set events '44951 trace name context forever, level 1024';
或者
c) Consider partitioning the LOB in a manner that will evenly distribute concurrent DML across multiple partitions。(分区减少段扩展的等待)

使用MSSM的:

a) As temporary workaround, manually add extra space to the LOB segment

ALTER TABLE <lob_table> MODIFY LOB (<column_name>) (allocate extent (size <extent size>));
或者
b) Consider partitioning the LOB in a manner that will evenly distribute concurrent DML across multiple partitions

我是用 alter system set events '44951 trace name context forever, level 1024'; 的方式解决的

解决前后对比:

未调整前:
Foreground Wait Events

Event                                      Waits  %Time -outs  Total Wait Time (s)  Avg wait (ms)  Waits /txn  % DB time
enq: HW - contention        243,239        0        267,844        1101       0.08      30.38
log file sync             2,879,162         0        88,962         31          1.00      10.09
SQL*Net message from dblink   107,467,710    0       59,295                        1          37.41     6.73

调整后Foreground Wait Events中未发现enq: HW - contention等待事件。

enq: TM - contention

三种可能引起改等待时间的情况:

  1.  执行DML 期间,为防止对与DML 相关的数据行进行修改,执行DML 的进程必须对该表获得TM 锁。若在获取TM 锁的过程中发生争用,则等待enq:TM-contention 事件。(通俗理解为一个是很多session在dml同一row)
  2. 在对父表进行update或delete时,子表上没有外键么有建立索引。如果外键上没有定义索引,对父表的主键执行 DML 操作时必须获得子表上的共享行排他表级锁(share row exclusive table lock)(也称为 share-subexclusive table lock,SSX)。此锁能够阻止其他事务对子表执行 DML 操作。SSX 锁在获得后立即释放。如果父表中有多个主键值被更新或删除,对每行执行 DML 操作时都需要对子表进行一次加锁及解锁操作。如果外键上定义了索引,则对父表的主键执行 DML 操作时只需获得子表上的行共享表级锁(row share table lock)(也称为 subshare table lock,SS)。此锁不允许其他事务排他地对子表加锁,但允许对父表及子表执行 DML 操作。
  3. dml和ddl操作发生阻塞

实验:

CREATE TABLE supplier
    (     supplier_id     number(10)     not null,
        supplier_name     varchar2(50)     not null,
        contact_name     varchar2(50),    
        CONSTRAINT supplier_pk PRIMARY KEY (supplier_id)
    );

INSERT INTO supplier VALUES (1, 'Supplier 1', 'Contact 1');
INSERT INTO supplier VALUES (2, 'Supplier 2', 'Contact 2');
COMMIT;

CREATE TABLE product
    (     product_id     number(10)     not null,
        product_name    varchar2(50)    not null,
        supplier_id     number(10)     not null,
        CONSTRAINT fk_supplier
          FOREIGN KEY (supplier_id)
         REFERENCES supplier(supplier_id)
         ON DELETE CASCADE
    );

INSERT INTO product VALUES (1, 'Product 1', 1);
INSERT INTO product VALUES (2, 'Product 2', 1);
INSERT INTO product VALUES (3, 'Product 3', 2);
COMMIT;

然后去执行几条相关的语句
User 1: DELETE supplier WHERE supplier_id = 1;
User 2: DELETE supplier WHERE supplier_id = 2;(现象HANG住)
User 3: INSERT INTO supplier VALUES (5, 'Supplier 5', 'Contact 5');(现象HANG住)

现在检查锁的情况

col event format a20
col type format a10
col object_name a15
col object_type a15
SELECT l.sid, s.blocking_session blocker, s.event, l.type, l.lmode, l.request, o.object_name, o.object_type
FROM v$lock l, dba_objects o, v$session s
WHERE UPPER(s.username) = UPPER('&User')
AND   l.id1        = o.object_id (+)
AND   l.sid        = s.sid
ORDER BY sid, type;
输入 user 的值:  test
原值    3: WHERE UPPER(s.username) = UPPER('&User')
新值    3: WHERE UPPER(s.username) = UPPER('test')

SID    BLOCKER EVENT                TYPE            LMODE    REQUEST
---------- ---------- -------------------- ---------- ---------- ----------
OBJECT_NAME     OBJECT_TYPE
--------------- ---------------
       144        158 enq: TM - contention TM                  3          0
SUPPLIER        TABLE

144        158 enq: TM - contention TM                  0          2
PRODUCT         TABLE

153            SQL*Net message from TM                  3          0
                       client
SUPPLIER        TABLE

SID    BLOCKER EVENT                TYPE            LMODE    REQUEST
---------- ---------- -------------------- ---------- ---------- ----------
OBJECT_NAME     OBJECT_TYPE
--------------- ---------------

153            SQL*Net message from TM                  3          0
                       client
PRODUCT         TABLE

153            SQL*Net message from TX                  6          0
                       client

SID    BLOCKER EVENT                TYPE            LMODE    REQUEST
---------- ---------- -------------------- ---------- ---------- ----------
OBJECT_NAME     OBJECT_TYPE
--------------- ---------------
       158        153 enq: TM - contention TM                  0          5
PRODUCT         TABLE

158        153 enq: TM - contention TM                  3          0
SUPPLIER        TABLE

已选择7行。

SQL>

能看到enq: TM - contention TM  等待事件

检查没有索引的外键

SELECT * FROM (
   SELECT c.table_name, cc.column_name, cc.position column_position
   FROM   user_constraints c, user_cons_columns cc
   WHERE  c.constraint_name = cc.constraint_name
   AND    c.constraint_type = 'R'
   MINUS
   SELECT i.table_name, ic.column_name, ic.column_position
   FROM   user_indexes i, user_ind_columns ic
   WHERE  i.index_name = ic.index_name
   )
ORDER BY table_name, column_position;
TABLE_NAME
------------------------------
COLUMN_NAME
--------------------------------------------------------------------------------

COLUMN_POSITION
---------------
PRODUCT
SUPPLIER_ID
              1

建立索引后:

CREATE INDEX fk_supplier ON product (supplier_id);

INSERT INTO supplier VALUES (6, 'Supplier 6', 'Contact 6');

INSERT INTO supplier VALUES (7, 'Supplier 7', 'Contact 7');

User 1: DELETE supplier WHERE supplier_id = 6;
User 2: DELETE supplier WHERE supplier_id = 7;
User 3: INSERT INTO supplier VALUES (8, 'Supplier 8', 'Contact 8');

相关的等待事件消失。

 

等待事件:enq: HW - contention和enq: TM - contention的更多相关文章

  1. oracle 11g enq: JI – contention等待事件

    最近使用物化视图同步的环境在大量刷新的时候频繁出现enq: JI – contention等待事件,经查: JI enqueue is acquired in exclusive mode on th ...

  2. ORACLE等待事件:enq: TX - row lock contention

    enq: TX - row lock contention等待事件,这个是数据库里面一个比较常见的等待事件.enq是enqueue的缩写,它是一种保护共享资源的锁定机制,一个排队机制,先进先出(FIF ...

  3. 【等待事件】序列等待事件总结(enq: SQ - contention、row cache lock、DFS lock handle和enq: SV - contention)

    [等待事件]序列等待事件总结(enq: SQ - contention.row cache lock.DFS lock handle和enq: SV -  contention) 1  BLOG文档结 ...

  4. enq: TX - row lock contention“等待事件的处理

      enq: TX - row lock contention“等待事件的处理   session1: SQL> conn scott/triger Connected. SQL> CRE ...

  5. enq: TM - contention一例

    今天下午,有台服务器出现异常,响应特别慢,io等待奇高,awr top 5事件如下: 经回查ash,找到了造成这些事件的sql语句,如下: select * from v$active_session ...

  6. truncate表hang住(等待时间较长),出现enq:RO fast object reuse等待事件

    有一个应用truncate表等待了一晚上,一个定时任务,跑了几年了,今天早上来发现昨晚没有执行完成,hang住了,查询发现等待事件 fast object reuse. 10.2.0.4的库 Bug ...

  7. Analyzing 'enq: HW - contention' Wait Event (Doc ID 740075.1)

    Analyzing 'enq: HW - contention' Wait Event (Doc ID 740075.1) In this Document   Symptoms   Cause   ...

  8. 【等待事件】等待事件系列(5.1)--Enqueue(队列等待)

    [等待事件]等待事件系列(5.1)--Enqueue(队列等待)   1  BLOG文档结构图   2  前言部分   2.1  导读和注意事项 各位技术爱好者,看完本文后,你可以掌握如下的技能,也可 ...

  9. Oracle常见的几种等待事件

    1. CPU time CPU time其实不是真正的等待事件.是衡量CPU是否瓶颈的一个重要指标.一般来讲,一个良好的系统,CPU TIME 应该排在TOP 5 TIME Event的最前面. 当然 ...

随机推荐

  1. unity tips

    1.在unity 的mecanim中,如果一个动画指向两个或两个以上的动画,那么在inspector中,transitions中可以看到所有的过渡路径,这些路径是有先后顺序的.

  2. Oracle SQL函数之数学函数

    Oracle SQL函数之数学函数 ABS(x) [功能]返回x的绝对值 [参数]x,数字型表达式 [返回]数字 SQL> SELECT ABS(),ABS(-) FROM DUAL; ABS( ...

  3. WIN7 Net Configuration Assistant打不开

     转自  http://www.cnblogs.com/caojie0432/archive/2013/07/30/3225230.html  作者:db_suploc 今天在安装oracle10g的 ...

  4. global中拦截404错误的实现方法

    1. void Application_Error(object sender, EventArgs e) { if(Context != null) { HttpContext ctx = Http ...

  5. android 补间动画

    android开发过程中,为了更好的展示应用程序,应用程序添加动画,能够很好地实现这个功能.如果动画中的图像变化有一定的规律,可以采用自动生成图像的方式来生成动画,例如图像的移动.旋转.缩放等.自动生 ...

  6. Failed to create the Java Virtual Machine (Myeclipse或者eclipse启动报错)

    把某几个值改为原来的0.5倍就ok了(我就这么解决的)   eclipse.ini如下:   -startupplugins/org.eclipse.equinox.launcher_1.2.0.v2 ...

  7. ubuntu安装体验

    本文记录一下昨晚及今天安装ubuntu系统的过程及体验 2016年6月13日09:36:11 更新 今天才有发现原来自己有个没填的坑 = = 那次安乌班图后第一感觉是很好用,新鲜了好几天,但是很快,新 ...

  8. delphi按钮控件的default属性

    delphi按钮控件的default属性用于设置默认命令按钮,.设置为true时,按[Enter键]相当于用鼠标单击了该按钮 .窗口中如果有多个按钮的default是true的话,就根据tabinde ...

  9. windows内存管理方式以及优缺点

    Windows内存管理方式:页式管理,段式管理,段页式管理 页式管理 将各进程的虚拟空间(逻辑地址)划分为若干个长度相等的页,业内管理把内存空间(物理内存)按照页的大小划分为片或者页面,从而实现了离散 ...

  10. OneNote Count

    用OneNote的时候,某个分区的笔记多的话,想数一下一共有多少笔记是个麻烦的活儿. OneNote没有自带这功能 于是写了个C#的小程序实现这功能 https://github.com/02xiao ...