时间:2021-07-01 10:21:17 帮助过:2人阅读
DBMS_OUTPUT ----调试PL/SQL,DEBUG
DBMS_SQL ----------执行动态SQL PL\SQL
dbms_job ------管理JOB
dbms_system -------跟踪用户,SESSION
dbms_transaction -------- 管理SQL事务
原则上,过程和包不允许独立编写,必须封装在包中。
与sybase不同, out 型的参数必须传入,不能缺省。
函数虽可定义out型变量,但不建议如此应用。当且仅当所实现只返回一个(含预估未来)基础数据类型(字符、数字、布尔、日期)的时候才将此功能申明为函数。其它情况下都申明为方法。
在包头有申明的过程在定义out型变量时,建议用一个结构体。目的是在未来扩展输出参数时,无须修改已修改的代码。
过程或函数定义基础类型参数时,varchar2, number 等即可。没必要使用用户自定义类型如:sys_type.u_acctno%type 或者kna_accs.acctno%type;
对于out 或in out 参数定义时 nocopy 关键字的意义:
不加nocopy传递的是复制(by value),否则传递的是地址(by reference)。
传递地址对大型结构或大数据时可以提高效率。
在出现异常时,不加nocopy的情况下,外层程序读到的仍然是原值。加nocopy的情况下,若有修改,外层程序读到的是修改之后的值。因此使用nocopy 时,需要确保在被调用过程中处理所有的异常。
Sybase需要用Begin tran来控制是否需要手动commit或rollback; Oralce无此语句,但有一开关Set autocommit OFF/ON;将commit或rollback的权力交给用户。默认关闭,即由用户来控制事务的提交或回滚;也就没有事务积数器这一全局变量(@@transcount)。
Alter … create …等都属于DDL语句。
由于Oracle执行DDL语句时会自动调用Commit,因此禁止在业务程序里使用DDL语句。
PRAGMA AUTONOMOUS_TRANSACTION;
可在某个子过程或函数中单独申明一段事务,有关其涉及到的业务逻辑处理,由自身commit或rollback,不会影响到主程序已处理的事务。
注意:函数中若有事务语句,必须申明独立事务。即:若无特殊情况,涉及事务处理都不写成函数,都应写成过程。
相关应用:生成流水号、记录密码错误等。
Sybase由系统控制异常,Oracle则可主动捕捉异常。每当PL/SQL违背了ORACLE原则或超越了系统依赖的原则就会隐式的产生内部异常。
异常分为两类:分为系统异常和用户自定义异常。用户定义异常则需要显现的抛出。
抛出方式 |
异常类型 |
通过PL/SQL运行引擎 |
系统内部异常 |
使用RAISE语句 |
用户定义异常 |
调用RAISE_APPLICATION_ERROR存储过程 |
用户定义异常 |
异常处理是用来处理正常执行过程中未预料的事件。如果PL/SQL程序块一旦产生异常而又没有指出如何处理时,程序会自动终止。
系统内部异常分为:预定义的内部异常和未定义内部异常。未定义内部异常只能在Other部分捕捉。
常见预定义异常 |
sqlerrm |
sqlcode |
描述 |
no_data_found |
ora-01403 |
+100 |
Select into无符合条件的记录 |
too_many_rows |
ora-01422 |
-1422 |
Select into符合条件的记录有多条 |
dup_val_on_index |
ora-00001 |
-1 |
违反唯一约束 |
value_error |
ora-06502 |
-6502 |
发生算数、转换、截断或大小约束错误 |
storage_error |
ora-06500 |
-6500 |
内存溢出 |
zero_divide |
ora-01476 |
-1476 |
除数为零 |
case_not_found |
ora-06592 |
-6530 |
无匹配when子句也无默认else子句 |
cursor_already_open |
ora-06511 |
-6511 |
试图打开已经打开的游标 |
timeout_on_resource |
ora-00051 |
-51 |
等待某一资源,超时 |
access_into_null |
ora-06530 |
-6530 |
试图给未初始化对象的属性赋值 |
invalid_cursor |
ora-01001 |
-1001 |
游标操作错误,如:关闭未打开 |
login_denied |
ora-01017 |
-1017 |
登录时用户名或密码非法 |
program_error |
ora-06501 |
-6501 |
PL/SQL内部错误 |
rowtype_mismatch |
ora-06504 |
-6504 |
赋值变量与游标返回变量不兼容 |
self_is_null |
ora-30625 |
-30625 |
|
sys_invalid_rowid |
ora-01410 |
-1410 |
字符串转化成rowid失败 |
subscript_beyond_count |
ora-06533 |
-6533 |
|
subscript_outside_limit |
ora-06532 |
-6532 |
|
collection_is_null |
ora-06531 |
-6531 |
|
invalid_number |
ora-01722 |
-1722 |
字符串转化成数字失败 |
not_logged_on |
ora-01012 |
-1012 |
未连接数据库前访问数据 |
当异常发生时,若本语句块未做异常处理,控制将转到或传播到外层语句块的异常处理部分。直到外层捕捉到异常,则运行捕捉异常后语句块;若外层块没有该异常的处理程序则传播到调用环境,并立即终止。
总原则:
任何异常处理必须有Others分支
1.SELECT INTO 语句:
常见no_data_found和too_many_rows异常,也就是说要保证查询结果有且只有一条数据才不会触发入上异常。因此不能用 SQL%Rowcount来判断行数。
若where 条件中包括物理上的唯一索引,则可以不对too_many_rows进行处理。
另:无group by 子句的,SELECT Count(*)、sum、max等聚合函数 INTO 能保证有且仅有一条数据,因此可以不对too_many_rows和 no_data_found进行处理。
2.INSERT 语句:
若插入的表存在唯一索引或主键,则需要判断 dup_val_on_index 异常;
空值异常在 Others 里处理 (ORA-01400);
能用 SQL%Rowcount 来判断行数。
3.Update语句:
若被修改的列是唯一索引所在列,则需要判断 dup_val_on_index 异常;
空值异常在 Others 里处理 (ORA-01400);
能用 SQL%Rowcount 来判断行数。
4.DELETE 语句:
处理Other异常即可,无须其它异常处理
能用 SQL%Rowcount 来判断行数。
为了程序结构清晰,应避免异常处理语句中出现异常嵌套;也就是说,异常处理中不应该再编写可能再次出现异常的语句如如:select、insert、delete、update等语句。可写成子程序或跳出此语句块后再处理。
不同于Sybase,临时表必须事先创建好。
临时表独立于每个session,也就是说,A连接不可能访问到B连接临时表中的数据。临时表有两种创建形式:
commit preserve rows,随着事务提交保存。数据一直存储,直到当前session断开。因此,在此种情况下,去维护临时表的表结构是行不通的。
commit delete rows,随着事务提交时清空。若只是用于中间计算,可用此种形式。
小技巧:临时表所有字段均为空,以便适用于不同场景。
也称关联数组、索引表。是普通集合的扩展,可以用健值获得数组中的值。
优点,键值唯一,访问速度快。
Type t_array is table of varchar2(20)
L_array t_array := {‘John’, ‘Susan’};
Dbms_Output.Put_Line(l_Array(2)); -- 下标
Type t_array_hash is table of varchar2(20) index by varchar2(4);
L_array_hash t_array_hash;
L_array_hash(‘0001’) := ‘John’;
L_array_hash(‘0002’) := ‘Susan’;
Dbms_Output.Put_Line(l_Array_Hash(‘0001‘)); -- 键值
如有下表(userid varchar2(5) is primary key, userna varchar2(20), brchno varchar2(6) )
TYPE t_User_Rec IS RECORD( -- 先对主键外字段建立“记录型”用户自定义类型
userna VARCHAR2(20),
brchno VARCHAR2(6)
);
-- 在结构上建立哈希表
TYPE t_User_tab IS TABLE OF t_User_Rec INDEX BY VARCHAR2(5);
l_User_Rec t_User_Rec; --申明记录变量
l_User_tab t_User_tab; --申明表变量
l_Userna varchar2(20); --普通变量
-- 哈希表插入记录
l_User_Rec.userna = ‘李大照’; l_User_Rec.brchno = ‘010201’;
l_User_tab(‘用户1’) := l_User_Rec; -- 记录1
l_User_Rec.userna = ‘王虎’; l_User_Rec.brchno = ‘019801’;
l_User_tab(‘00002’) := l_User_Rec; -- 记录2
-- 访问记录
l_Userna = l_User_tab(‘00002’).Userna;
如上表需增加维度行号(bankno varchar2(12) is primary key, userid varchar2(5) is primary key,…)
…
-- 需要增加的工作是对一维哈希表再建立一层哈希表
TYPE t_Bank_User_tab IS TABLE OF t_User_tab INDEX BY VARCHAR2(12);
l_User_Rec t_User_Rec; --申明记录变量
l_bank_user t_Bank_User_tab --申明二维表变量
l_Userna varchar2(20); --普通变量
-- 哈希表插入记录
l_User_Rec.userna = ‘李大照’; l_User_Rec.brchno = ‘010201’;
l_User_tab(‘武汉分行’)(‘用户1’) := l_User_Rec; -- 记录1
l_User_Rec.userna = ‘王虎’; l_User_Rec.brchno = ‘019801’;
l_User_tab(‘常洲分行’)(‘00002’) := l_User_Rec; -- 记录2
-- 访问记录
l_Userna = l_User_tab(‘常洲分行’)(‘00002’).Userna;
数据库基础知识
标签: