在Oracle数据库的PLSQL开发工作中,dbms_lock.sleep()是一个用于实现执行延时的内置存储过程,它可以让当前会话暂停指定的时间后再继续执行后续逻辑,在很多业务场景中都能发挥重要作用。

dbms_lock.sleep()基本语法
这个存储过程属于dbms_lock包,基本调用语法如下:
-- 暂停指定秒数,参数类型为NUMBER,支持小数,最小精度为百分之一秒 DBMS_LOCK.SLEEP (seconds IN NUMBER);
需要注意的是,调用这个存储过程需要先获取对应的执行权限,普通用户默认没有权限,需要管理员执行授权操作:
-- 管理员执行授权,将dbms_lock包的执行权限赋予指定用户 GRANT EXECUTE ON DBMS_LOCK TO 用户名;
常见使用场景
1. 控制任务执行频率
当编写定时执行的PLSQL任务时,如果需要控制两次执行之间的间隔,就可以使用这个存储过程。比如每分钟执行一次数据同步逻辑,执行完同步后暂停60秒再进入下一次循环。
2. 等待依赖资源就绪
如果当前逻辑需要依赖其他会话的操作结果,比如等待某个临时表被写入数据,或者等待某个锁被释放,可以先尝试查询状态,如果未就绪就调用sleep暂停一段时间再重试。
3. 模拟业务耗时
在测试环境模拟真实业务的执行耗时,比如模拟一个业务操作需要3秒的处理时间,就可以在测试代码中加入对应的sleep调用,验证系统的超时处理逻辑是否正常。
使用注意事项
- sleep期间当前会话会处于等待状态,不会释放已经持有的锁资源,所以如果长时间暂停可能会导致其他会话等待锁超时。
- 参数的单位是秒,如果需要暂停毫秒级的时间,可以传入小数,比如暂停500毫秒就传入0.5。
- 在Oracle 18c及之后的版本中,也可以使用
DBMS_SESSION.SLEEP替代,功能基本一致,且不需要额外授权。
完整实例演示
下面是一个完整的PLSQL块示例,实现了每10秒查询一次当前系统时间,总共执行3次的逻辑:
DECLARE
v_count NUMBER := 1; -- 循环计数器
v_current_time VARCHAR2(30); -- 存储当前时间
BEGIN
WHILE v_count <= 3 LOOP
-- 获取当前系统时间
SELECT TO_CHAR(SYSDATE, 'YYYY-MM-DD HH24:MI:SS')
INTO v_current_time
FROM DUAL;
-- 输出当前时间和循环次数
DBMS_OUTPUT.PUT_LINE('第' || v_count || '次执行,当前时间:' || v_current_time);
-- 如果不是最后一次循环,暂停10秒
IF v_count < 3 THEN
DBMS_LOCK.SLEEP(10);
END IF;
v_count := v_count + 1;
END LOOP;
END;
/
执行上述代码后,会先输出第一次的时间,然后暂停10秒输出第二次,再暂停10秒输出第三次,符合预期的执行效果。
替代方案说明
如果因为权限问题无法使用dbms_lock.sleep(),也可以自己编写简单的延时逻辑,不过精度会差一些,示例代码如下:
-- 简单的延时函数,精度较低,适合对延时精度要求不高的场景
CREATE OR REPLACE PROCEDURE custom_sleep(p_seconds IN NUMBER) IS
v_start_time NUMBER;
BEGIN
v_start_time := DBMS_UTILITY.GET_TIME; -- 获取当前时间,单位是百分之一秒
-- 循环等待直到时间到达
WHILE (DBMS_UTILITY.GET_TIME - v_start_time) / 100 < p_seconds LOOP
NULL; -- 空操作,仅用于等待
END LOOP;
END;
/
不过这种自定义方式的CPU占用会比内置的sleep高,所以优先还是使用内置的dbms_lock.sleep()存储过程。
dbms_lock.sleepOracle存储过程PLSQL延时数据库锁PLSQL编程修改时间:2026-07-05 10:15:24