#DBMS_UTILITY
DBMS_UTILITY包提供了一组内置的实用存储过程/函数。
# ACTIVE_INSTANCES
TYPE INSTANCE_RECORD IS RECORD (inst_number NUMBER, inst_name VARCHAR(64));
TYPE INSTANCE_TABLE IS TABLE OF INSTANCE_RECORD;
DBMS_UTILITY.ACTIVE_INSTANCES (
instance_table OUT INSTANCE_TABLE,
instance_count OUT NUMBER);
ACTIVE_INSTANCES过程用于查看数据库活跃实例的信息,返回活跃实例的个数以及每个活跃实例的ID和名称。该过程不适用于分布式部署。
参数 | 描述 |
---|---|
instance_table | 包含活动实例编号和名称的列表。当没有实例启动时,列表为空 |
instance_count | 活跃实例个数 |
示例(单机、共享集群部署)
SET serveroutput ON
DECLARE
inst_record DBMS_UTILITY.INSTANCE_RECORD;
inst_tab DBMS_UTILITY.INSTANCE_TABLE;
inst_cnt NUMBER;
BEGIN
DBMS_UTILITY.ACTIVE_INSTANCES(inst_tab, inst_cnt);
FOR i IN 1..inst_cnt LOOP
DBMS_OUTPUT.PUT_LINE(inst_tab(i).inst_number || ' <=> ' || inst_tab(i).inst_name);
END LOOP;
END;
/
---result
1 <=> yasdb
# COMMA_TO_TABLE
TYPE LNAME_ARRAY IS TABLE of VARCHAR(32000) INDEX BY BINARY_INTEGER;
TYPE UNCL_ARRAY IS TABLE OF VARCHAR(32000) INDEX BY BINARY_INTEGER;
DBMS_UTILITY.COMMA_TO_TABLE (
list IN VARCHAR2,
tablen OUT BINARY_INTEGER,
tab OUT uncl_array);
DBMS_UTILITY.COMMA_TO_TABLE (
list IN VARCHAR2,
tablen OUT BINARY_INTEGER,
tab OUT lname_array);
COMMA_TO_TABLE过程用于将使用逗号分隔的name字符串转换为数组。该过程不适用于分布式部署。
参数 | 描述 |
---|---|
list | 逗号分隔的'name'列表字符串。根据tab类型不同,name字符串支持以下形式: * tab为UNCL_ARRAY类型时,name字符串形式为 identifier[.identifier[.identifier]]@dblink * tab为LNAME_ARRAY类型时,name字符串形式为 identifier[.identifier]* |
tablen | 分割的name数 |
tab | name列表字符串转换成的关联数组 |
示例(单机、共享集群部署)
DECLARE
l_list VARCHAR(4000) := ' c , a.b.c , dd@dd@"k" ';
l_count BINARY_INTEGER;
l_array DBMS_UTILITY.UNCL_ARRAY;
BEGIN
-- 将逗号分隔的字符串转换为表
DBMS_UTILITY.COMMA_TO_TABLE(list => l_list, tablen => l_count, tab => l_array);
-- 输出转换后的表的元素数
DBMS_OUTPUT.PUT_LINE('Number of elements: ' || TO_CHAR(l_count));
DBMS_OUTPUT.PUT_LINE('1:' || l_array(1) || 'end' );
DBMS_OUTPUT.PUT_LINE('2:' || l_array(2) || 'end');
DBMS_OUTPUT.PUT_LINE('3:' || l_array(3) || 'end' );
END;
/
--result
Number of elements: 3
1: c end
2: a.b.c end
3: dd@dd@"k" end
# CURRENT_INSTANCE
DBMS_UTILITY.CURRENT_INSTANCE()
RETURN NUMBER;
CURRENT_INSTANCE函数用于获取当前连接的实例号。
示例
SELECT DBMS_UTILITY.CURRENT_INSTANCE() FROM dual;
CURRENT_INSTANCE
----------------
1
# DB_VERSION
dbms_utility.db_version (
version OUT VARCHAR2,
compatibility OUT VARCHAR2);
DB_VERSION过程用于获取数据库的版本信息。
参数 | 描述 |
---|---|
version | 一个字符串,表示数据库的内部软件版本 |
compatibility | 保留参数,仅用于兼容,参数值无实际影响 |
示例
SET serveroutput ON
DECLARE
ver CHAR(16);
compat CHAR(12);
BEGIN
DBMS_UTILITY.DB_VERSION(ver, compat);
DBMS_OUTPUT.PUT_LINE('Version: ' || ver ||' Compatible: ' || compat);
END;
/
--- result
Version: 23.2.3.51 Compatible:
# EXEC_DDL_STATEMENT
DBMS_UTILITY.EXEC_DDL_STATEMENT (
parse_string IN VARCHAR2
);
EXEC_DDL_STATEMENT过程用于执行DDL语句。该过程不适用于分布式部署。
参数 | 描述 |
---|---|
parse_string | 待执行的语句 |
示例
DECLARE
BEGIN
DBMS_UTILITY.EXEC_DDL_STATEMENT('CREATE TABLE UTILITY_TABLE(A INT, B INT)');
END;
/
SELECT A,B FROM UTILITY_TABLE;
--result
A B
------------ ------------
# FORMAT_CALL_STACK
DBMS_UTILITY.FORMAT_CALL_STACK()
RETURN VARCHAR;
FORMAT_CALL_STACK函数返回当前过程体调用栈的格式化输出,无参数,'()'可以省略,返回值最大2000个字节。
示例
CREATE OR REPLACE PROCEDURE PROC_CALL AS
BEGIN
DBMS_OUTPUT.PUT_LINE(DBMS_UTILITY.FORMAT_CALL_STACK);
END;
/
CALL PROC_CALL;
--result
----- PL/SQL Call Stack ----
object handle line number object name
0x7fffe2e39010 3 procedure REGRESS.PROC_CALL
0x7fffdfb3bd90 1 anonymous block
# FORMAT_ERROR_BACKTRACE
DBMS_UTILITY.FORMAT_ERROR_BACKTRACE()
RETURN VARCHAR2;
FORMAT_ERROR_BACKTRACE函数显示引发异常的位置,最大长度为4000字节。
示例
SET serveroutput ON
CREATE OR REPLACE PROCEDURE format_error_backtrace_p0 IS
str VARCHAR(4000);
BEGIN
str := 1/0;
DBMS_OUTPUT.PUT_LINE('Hello, this is my first stored procedure!');
END format_error_backtrace_p0;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_p1 IS
BEGIN
format_error_backtrace_p0();
END format_error_backtrace_p1;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_p2 IS
BEGIN
format_error_backtrace_p1();
END format_error_backtrace_p2;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_p3 IS
BEGIN
format_error_backtrace_p2();
END format_error_backtrace_p3;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_p4 IS
BEGIN
format_error_backtrace_p3();
END format_error_backtrace_p4;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_p5 IS
BEGIN
format_error_backtrace_p4();
END format_error_backtrace_p5;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_top_nolog IS
BEGIN
format_error_backtrace_p5();
END format_error_backtrace_top_nolog;
/
CREATE OR REPLACE PROCEDURE format_error_backtrace_top_logging IS
BEGIN
format_error_backtrace_p5();
EXCEPTION
WHEN others THEN
DBMS_OUTPUT.PUT_LINE('Error_Backtrace...' || CHR(10) ||
DBMS_UTILITY.FORMAT_ERROR_BACKTRACE());
DBMS_OUTPUT.PUT_LINE( '----------' );
END format_error_backtrace_top_logging;
/
exec format_error_backtrace_top_logging;
--- result
Error_Backtrace...
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_P0", line 4
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_P1", line 3
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_P2", line 3
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_P3", line 3
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_P4", line 3
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_P5", line 4
YAS-04015 at "SYS.FORMAT_ERROR_BACKTRACE_TOP_LOGGING", line 3
----------
# FORMAT_ERROR_STACK
DBMS_UTILITY.FORMAT_ERROR_STACK()
RETURN VARCHAR;
FORMAT_ERROR_STACK函数返回当前错误栈的格式化输出,无参数,'()'可以省略,返回值最大2000个字节。
示例
DECLARE
N INT;
BEGIN
N := 10 / 0;
EXCEPTION
WHEN OTHERS THEN
DBMS_OUTPUT.PUT_LINE(DBMS_UTILITY.FORMAT_ERROR_STACK);
END;
/
--result
YAS-00011 divided by zero
# GET_ENDIANNESS
DBMS_UTILITY.GET_ENDIANNESS()
RETURN NUMBER;
GET_ENDIANNESS函数用于获取当前数据库平台的字节序,返回值为1表示大端字节序,返回值为2表示小端字节序。
示例
SELECT DBMS_UTILITY.GET_ENDIANNESS() FROM dual;
# GET_HASH_VALUE
DBMS_UTILITY.GET_HASH_VALUE (
NAME IN VARCHAR,
BASE IN NUMBER,
HASH_SIZE IN NUMBER)
RETURN NUMBER;
GET_HASH_VALUE函数为字符串计算一个落在给定范围内的哈希值,该范围是[BASE, BASE + HASH_SIZE - 1]
。
参数 | 描述 |
---|---|
NAME | 需要计算哈希值的字符串,不允许为NULL或空字符串 |
BASE | 哈希值的起始值 |
HASH_SIZE | 哈希值的大小 |
使用说明:
- BASE和HASH_SIZE的取值范围是[-231, 231 - 1]。
- BASE取值如果是浮点数,会四舍五入取整。
- HASH_SIZE取值不能是0或浮点数。
示例
SELECT DBMS_UTILITY.GET_HASH_VALUE('abc', 1000, 2048) FROM dual;
# GET_PARAMETER_VALUE
DBMS_UTILITY.GET_PARAMETER_VALUE (
parnam IN VARCHAR2,
intval IN OUT BINARY_INTEGER,
strval IN OUT VARCHAR2,
listno IN BINARY_INTEGER DEFAULT 1)
RETURN BINARY_INTEGER;
GET_PARAMETER_VALUE函数将获取系统参数的值,对应值可以在V$PARAMETER上查到。
如果参数类型为INTEGER/BOOL时,返回值为0。如果类型是VARCHAR,则返回是1。
参数 | 描述 |
---|---|
param | 参数名称 |
intval | 如果参数值是INTEGER/BOOL类型,则返回对应值。如果参数值是字符串类型,则返回对应值的长度 |
strval | 如果参数值是INTEGER/BOOL类型,则返回值是NULL。如果参数值是字符串类型,则返回对应值 |
listno | 保留参数,仅用于兼容,参数值无实际影响 |
示例
DECLARE
-- 参数名、整数值、字符串值及参数类型
parnam VARCHAR2(256) := 'DSINTERVAL_FORMAT'; -- 参数名称
intval BINARY_INTEGER; -- 可存储整数参数的值或字符串参数值的长度
strval VARCHAR2(256); -- 可存储字符串参数的值
partyp BINARY_INTEGER; -- 参数类型标识符
BEGIN
-- 使用GET_PARAMETER_VALUE函数检索参数值及类型
partyp := DBMS_UTILITY.GET_PARAMETER_VALUE(parnam, intval, strval);
-- 输出参数值
DBMS_OUTPUT.PUT('Parameter value is: ');
IF partyp = 1 THEN
DBMS_OUTPUT.PUT_LINE(strval); -- 如果是字符串参数,输出字符串值
ELSE
DBMS_OUTPUT.PUT_LINE(TO_CHAR(intval)); -- 如果是整数参数,输出整数值
END IF;
-- 如果是字符串参数,还可以输出该字符串的长度
IF partyp = 1 THEN
DBMS_OUTPUT.PUT('Parameter value length is: ');
DBMS_OUTPUT.PUT_LINE(intval); -- 输出字符串值的长度
END IF;
-- 输出参数类型(整数或字符串)
DBMS_OUTPUT.PUT('Parameter type is: ');
IF partyp = 1 THEN
DBMS_OUTPUT.PUT_LINE('string');
ELSE
DBMS_OUTPUT.PUT_LINE('integer');
END IF;
END;
/
--result
Parameter value is: dd hh24:mi:ss.ff
Parameter value length is: 16
Parameter type is: string
# GET_SQL_HASH
DBMS_UTILITY.GET_SQL_HASH (
name IN VARCHAR2,
hash OUT RAW,
pre10ihash OUT NUMBER)
RETURN NUMBER;
GET_SQL_HASH函数使用MD5算法计算给定字符串的哈希值。
参数 | 描述 |
---|---|
name | 要进行哈希计算的字符串 |
hash | 用于存储所有返回16字节哈希值的可选字段 |
pre10ihash | 不支持此参数,返回结果为NULL |
示例
SET serveroutput ON
DECLARE
h VARCHAR(32);
n NUMBER;
x NUMBER;
BEGIN
x := DBMS_UTILITY.GET_SQL_HASH('select 1 from t1', h, n);
DBMS_OUTPUT.PUT_LINE('Return Value: ' || x);
DBMS_OUTPUT.PUT_LINE('Hash: ' || h);
DBMS_OUTPUT.PUT_LINE('Pre10iHash: ' || TO_CHAR(n));
END;
/
--- result
Return Value: 1334230816
Hash: 1E7E2F21208F80F4BEB552F320BF864F
Pre10iHash:
# GET_TIME
DBMS_UTILITY.GET_TIME()
RETURN NUMBER;
GET_TIME函数返回一个数值(本地时间戳)用于表示当前时间,以百分之一秒为单位。
该数值与系统当前时间相关,可能为负数。
示例
-- 获取当前时间
SELECT DBMS_UTILITY.GET_TIME() FROM dual;
-- 输出 200hsec = 2 sec
SET serveroutput ON;
DECLARE
start_time NUMBER;
end_time NUMBER;
BEGIN
start_time:= DBMS_UTILITY.GET_TIME();
DBMS_LOCK.SLEEP(2);
end_time:= DBMS_UTILITY.GET_TIME() - start_time;
DBMS_OUTPUT.PUT_LINE('elapsed = ' || end_time || 'hsecs');
END;
/
SET serveroutput OFF;
# IS_BIT_SET
DBMS_UTILITY.IS_BIT_SET (
R IN RAW,
N IN NUMBER)
RETURN NUMBER;
IS_BIT_SET函数检查给定的RAW变量对应位置的比特(bit)是否被设置。
参数 | 描述 |
---|---|
R | 需要检查的RAW变量 |
N | 需要检查的位置,N的取值只能为正整数且小于RAW变量的比特位长度 |
示例
SELECT DBMS_UTILITY.IS_BIT_SET(HEXTORAW('AB'), 1) FROM DUAL;
# IS_CLUSTER_DATABASE
DBMS_UTILITY.IS_CLUSTER_DATABASE()
RETURN BOOLEAN;
IS_CLUSTER_DATABASE函数判断数据库是否处于共享集群模式,如果是就返回TRUE,否则返回FALSE。
示例
SET serveroutput ON;
BEGIN
IF DBMS_UTILITY.IS_CLUSTER_DATABASE THEN
DBMS_OUTPUT.PUT_LINE('TRUE');
ELSE
DBMS_OUTPUT.PUT_LINE('FALSE');
END IF;
END;
/
SET serveroutput OFF;
# NAME_RESOLVE
DBMS_UTILITY.NAME_RESOLVE (
name IN VARCHAR,
context IN NUMBER,
schema OUT VARCHAR,
part1 OUT VARCHAR,
part2 OUT VARCHAR,
dblink OUT VARCHAR,
part1_type OUT NUMBER,
object_number OUT NUMBER);
本过程的输入形式等同于NAME_TOKENIZE过程。本过程尝试将name解析成a,b,c,dblink四个部分,并根据context指定的值,确认a,b,c,dblink的实际含义。
参数 | 描述 |
---|---|
name | 对象名,格式为[[a.]b.]c[@d],其中a、b、c均为SQL对象,d为dblink |
context | [0,9]范围的整数,指定解析的模式: * 0 - table:表 * 1 - function/procedure/package:函数/过程/包 * 2 - sequence:序列 * 3 - trigger:触发器 * 7 - type:类型 * 9 - index:索引 |
schema | 对象的schema,如果未指定,名字由对象所属schema决定 |
part1 | 名字的第一部分,此名称的类型指定为part1_type(同义词或包) |
part2 | 如果该值为非NULL,则表示子程序名。如果part1为非NULL,则part2指示的子程序位于part1所指示的包内;如果part1为NULL,则子程序是顶级子程序 |
dblink | 如果该值为非NULL,则表示数据库链接被指定为name的一部分,或者name是一个同义词,该同义词解析为具有数据库链接的内容。在这种情况下,如需进一步进行名称转换(即执行远程节点上的NAME_RESOLVE过程),应调用DBMS_UTILITY此远程节点上的NAME_RESOLVE过程 |
part1_type | 第一部分的类型 |
object_number | 对象ID |
过程使用规则:
- 当对象名[[a.]b.]c[@d]中的dblink不为空时,NAME_RESOLVE退化为NAME_TOKENIZE,仅解析并输出schema,part1,part2和dblink四个部分。
- 当对象名[[a.]b.]c[@d]中的dblink为空时,会从a,b和c三个部分中解析出schema,objectName,c部分仅当context为1时,对package对象有效。
- 当对象名[[a.]b.]c[@d]中仅有a不为空时,则a为objectName,schema为当前用户名;当对象名[[a.]b.]c[@d]中a,b都不为空时,则a为schema,b为objectName。随后根据c部分的context值获取对应类型的对象oid,并通过object_number参数返回对应的oid、part1_type参数返回对应的对象类型,若无法获取到对应类型的对象则报错。
示例
-- 以sales用户的area表为例
DECLARE
l_schema VARCHAR2(30);
l_part1 VARCHAR2(30);
l_part2 VARCHAR2(30);
l_dblink VARCHAR2(30);
l_part1_type NUMBER;
l_object_number NUMBER;
BEGIN
DBMS_UTILITY.NAME_RESOLVE('area', 0, l_schema, l_part1, l_part2, l_dblink, l_part1_type, l_object_number);
DBMS_OUTPUT.PUT_LINE('Schema: ' || l_schema);
DBMS_OUTPUT.PUT_LINE('Part1: ' || l_part1);
DBMS_OUTPUT.PUT_LINE('Part2: ' || l_part2);
DBMS_OUTPUT.PUT_LINE('DB Link: ' || l_dblink);
DBMS_OUTPUT.PUT_LINE('Part1 Type: ' || l_part1_type);
DBMS_OUTPUT.PUT_LINE('Object Number: ' || l_object_number);
END;
/
--result
Schema: SALES
Part1: AREA
Part2:
DB Link:
Part1 Type: 1
Object Number: 2426
# NAME_TOKENIZE
DBMS_UTILITY.NAME_TOKENIZE (
name IN VARCHAR,
a OUT VARCHAR,
b OUT VARCHAR,
c OUT VARCHAR,
dblink OUT VARCHAR,
nextpos OUT BINARY_INTEGER);
NAME_TOKENIZE过程使用parser将输入identifier[.identifier[.identifier]][@dblink]
形式的name输出为a、b、c和dblink,其中dblink的形式为identifier[.identifier]*[@identifier]
。对于非双引号的identifier将转换成大写。
参数 | 描述 |
---|---|
name | 由sql identifier组成,形如scott.foo@dblink |
a | 第一部分token |
b | 第二部分token |
c | 第三部分token |
dblink | @之后的dblink部分 |
nextpos | parse终止的字符串下标,例如a.b c ,则nextpos为3 |
示例
DECLARE
name VARCHAR(20000) := 'a.d.c@rdb';
a VARCHAR(2000);
b VARCHAR(2000);
c VARCHAR(2000);
dblink VARCHAR(2000);
nextpos BINARY_INTEGER;
BEGIN
DBMS_UTILITY.NAME_TOKENIZE(
name => name,
a => a,
b => b,
c => c,
dblink => dblink,
nextpos => nextpos
);
DBMS_OUTPUT.PUT_LINE('A: ' || a);
DBMS_OUTPUT.PUT_LINE('B: ' || b);
DBMS_OUTPUT.PUT_LINE('C: ' || c);
DBMS_OUTPUT.PUT_LINE('DBLink: ' || dblink);
DBMS_OUTPUT.PUT_LINE('Nextpos: ' || nextpos);
END;
/
--result
A: A
B: D
C: C
DBLink: RDB
Nextpos: 9
# OLD_CURRENT_SCHEMA
DBMS_UTILITY.OLD_CURRENT_SCHEMA()
RETURN VARCHAR;
OLD_CURRENT_SCHEMA函数返回当前会话的SCHEMA。
示例
SELECT DBMS_UTILITY.OLD_CURRENT_SCHEMA() FROM dual;
# OLD_CURRENT_USER
DBMS_UTILITY.OLD_CURRENT_USER()
RETURN VARCHAR;
OLD_CURRENT_USER函数返回当前会话的USER。
示例
SELECT DBMS_UTILITY.OLD_CURRENT_USER() FROM dual;
# PORT_STRING
DBMS_UTILITY.PORT_STRING()
RETURN VARCHAR;
PORT_STRING函数返回操作系统的平台以及版本。
示例
SELECT DBMS_UTILITY.PORT_STRING() FROM dual;
# TABLE_TO_COMMA
DBMS_UTILITY.TABLE_TO_COMMA (
tab IN UNCL_ARRAY,
tablen OUT BINARY_INTEGER,
list OUT VARCHAR2);
DBMS_UTILITY.TABLE_TO_COMMA (
tab IN LNAME_ARRAY,
tablen OUT BINARY_INTEGER,
list OUT VARCHAR2);
TABLE_TO_COMMA过程将UNCL_ARRAY或者是LNAME_ARRAY类型的name数组转换成逗号分隔的字符串。该过程不适用于分布式部署。
该过程不会对tab的输入进行校验。
参数 | 描述 |
---|---|
tab | 'name'的关联数组 |
tablen | 数组长度 |
list | 返回的逗号分隔的'name'的字符串 |
示例(单机、共享集群部署)
DECLARE
-- 使用DBMS_UTILITY.LNAME_ARRAY类型
myArray DBMS_UTILITY.UNCL_ARRAY;
-- 用于存储转换后的字符串和数组长度
listOut VARCHAR(4000);
tabLen BINARY_INTEGER;
BEGIN
-- 向数组中添加数据
myArray(1) := '++++ ++++ +++ +++ ';
myArray(2) := '1111111100000000000001111111111111111';
-- 调用TABLE_TO_COMMA过程
DBMS_UTILITY.TABLE_TO_COMMA(
tab => myArray,
tablen => tabLen,
list => listOut
);
-- 输出结果
DBMS_OUTPUT.PUT_LINE('Comma-separated list: ' || listOut);
DBMS_OUTPUT.PUT_LINE('Number of elements: ' || tabLen);
END;
/
--result
iComma-separated list: ++++ ++++ +++ +++ ,1111111100000000000001111111111111111
Number of elements: 2