该脚本用于检查Oracle数据库中被锁定的数据库对象(表、索引等),并关联获取持有锁的会话信息,是定位死锁、锁阻塞问题,排查业务操作卡顿的核心工具。
一、sql脚本
sql
SELECT S.SID,
S.SERIAL#,
S.USERNAME,
S.SCHEMANAME,
S.OSUSER,
S.MACHINE,
S.TERMINAL,
S.PROGRAM,
O.OWNER,
O.OBJECT_NAME,
O.OBJECT_TYPE,
O.OBJECT_ID
FROM DBA_OBJECTS O, V$LOCKED_OBJECT L, V$SESSION S
WHERE O.OBJECT_ID = L.OBJECT_ID
AND S.SID = L.SESSION_ID;
二、脚本说明
1、使用场景
- 故障排查:业务操作卡顿、超时,或出现"ORA-00054: 资源正忙"错误时,定位锁持有会话;
- 日常巡检:定期检查锁状态,提前发现潜在的锁阻塞风险;
- 变更验证:批量数据操作后,确认是否存在残留锁;
- 性能优化:分析锁持有模式,优化业务SQL的锁竞争问题。
2、字段说明
| 字段名 | 说明 |
|---|---|
SID |
持有锁的会话ID,用于后续终止异常会话 |
SERIAL# |
会话序列号,与SID配合唯一标识会话 |
USERNAME |
持有锁的数据库用户名,定位业务归属 |
SCHEMANAME |
会话对应的Schema名称 |
OSUSER |
会话对应的操作系统用户,定位客户端来源 |
MACHINE |
会话所在的客户端主机名,定位锁的物理来源 |
PROGRAM |
会话对应的客户端程序(如应用服务器、PL/SQL Developer) |
OWNER |
被锁定对象的所有者 |
OBJECT_NAME |
被锁定的对象名称(如业务表、索引) |
OBJECT_TYPE |
被锁定对象的类型(如TABLE、INDEX) |
OBJECT_ID |
被锁定对象的唯一ID,用于关联查询 |
三、注意事项
查询结果有记录返回即表示存在锁持有情况,以下情况代表存在问题:
- 核心业务对象(如交易表、订单表)被长时间锁定(超过10分钟);
- 同一对象被多个会话持有锁,或存在锁等待链 (需结合
V$LOCK视图进一步排查); - 持有锁的会话处于非活跃状态(如客户端已断开,但会话未释放锁);
- 锁持有时间与业务操作预期不符(如简单查询持有锁超过1分钟)。
四、补充说明
1、异常的影响
- 业务阻塞:被锁定对象无法执行写入操作,导致业务操作超时、失败;
- 性能下降:锁竞争会增加CPU、内存消耗,引发数据库整体响应变慢;
- 死锁风险 :多个会话互相持有对方需要的锁时,会引发死锁(
ORA-00060错误),导致事务回滚; - 资源浪费:非活跃会话持有锁会占用数据库资源,无法被其他会话使用。
2、处理建议
-
验证锁的合理性:确认锁持有是否为正常业务操作(如批量数据更新),若为正常操作可等待任务完成;
-
终止异常会话 :若为异常锁(如客户端断开未释放的锁),可通过
SID和SERIAL#终止会话:sqlALTER SYSTEM KILL SESSION 'SID, SERIAL#' IMMEDIATE; -
优化业务SQL:减少锁持有时间(如缩短事务范围、避免全表锁定),使用行级锁替代表级锁;
-
排查锁等待链 :结合
V$LOCK视图定位锁阻塞的源头:sqlSELECT BLOCKING_SESSION, SID, TYPE, ID1, ID2 FROM V$LOCK WHERE BLOCK = 1; -
配置锁告警:对核心业务对象的锁持有时间配置告警(如超过5分钟触发告警)。