Oracle存储过程详解(⼀)存储过程创建语法:
create [or replace] procedure存储过程名(param1 in type,param2 out type)
as
变量1类型(值范围);
变量2 类型(值范围);
Begin
Select count(*) into变量1from表A where列名=param1;
If (判断条件) then
Select列名into变量2from表A where列名=param1;
Dbms_output。Put_line(‘打印信息’);
Elsif (判断条件) then
Dbms_output。Put_line(‘打印信息’);
Else
Raise 异常名(NO_DATA_FOUND);
End if;
Exception
When others then
Rollback;
End;
注意事项:
1. 存储过程参数不带取值范围,in表⽰传⼊,out表⽰输出
2. 变量带取值范围,后⾯接分号
3. 在判断语句前最好先⽤count(*)函数判断是否存在该条操作记录
4. ⽤select … into … 给变量赋值
5. 在代码中抛异常⽤ raise+异常名
已命名的异常:
命名的系统异常产⽣原因
ACCESS_INTO_NULL未定义对象
CASE_NOT_FOUND CASE 中若未包含相应的 WHEN ,并且没有设置ELSE 时
COLLECTION_IS_NULL集合元素未初始化
CURSER_ALREADY_OPEN游标已经打开
DUP_VAL_ON_INDEX唯⼀索引对应的列上有重复的值
INVALID_CURSOR在不合法的游标上进⾏操作
INVALID_NUMBER内嵌的 SQL 语句不能将字符转换为数字
NO_DATA_FOUND使⽤ select into 未返回⾏,或应⽤索引表未初始化的
TOO_MANY_ROWS执⾏ select into 时,结果集超过⼀⾏
ZERO_DIVIDE除数为 0
SUBSCRIPT_BEYOND_COUNT元素下标超过嵌套表或 VARRAY 的最⼤值
SUBSCRIPT_OUTSIDE_LIMIT使⽤嵌套表或 VARRAY 时,将下标指定为负数
VALUE_ERROR赋值时,变量长度不⾜以容纳实际数据
LOGIN_DENIED PL/SQL 应⽤程序连接到 oracle 数据库时,提供了不正确的⽤户名或密码NOT_LOGGED_ON
PL/SQL 应⽤程序在没有连接 oralce 数据库的情况下访问数据PROGRAM_ERROR
PL/SQL 内部问题,可能需要重装数据字典& pl./SQL系统包ROWTYPE_MISMATCH
宿主游标变量与 PL/SQL 游标变量的返回类型不兼容SELF_IS_NULL
使⽤对象类型时,在 null 对象上调⽤对象⽅法STORAGE_ERROR
运⾏ PL/SQL 时,超出内存空间SYS_INVALID_ID
⽆效的 ROWID 字符串TIMEOUT_ON_RESOURCE Oracle 在等待资源时超时
命名的系统异常
产⽣原因基本语法
1. 基本结构
2. SELECT INTO STATEMENT
将select查询的结果存⼊到变量中,可以同时将多个列存储多个变量中,必须有⼀条记录,否则抛出异常(如果没有记录抛出
NO_DATA_FOUND)
例⼦:
3. IF 判断
4. while 循环
5. 变量赋值
CREATE  OR  REPLACE PROCEDURE  存储过程名字
(
参数1 IN  NUMBER,
参数2 IN  NUMBER
) IS
变量1 INTEGER  :=0;
变量2 DATE;
BEGIN
--执⾏体
END  存储过程名字;
BEGIN
SELECT  col1,col2 into  变量1,变量2 FROM  typestruct where  xxx;
EXCEPTION
WHEN NO_DATA_FOUND THEN
xxxx;
END ;
IF V_TEST=1 THEN
BEGIN
do  something
END ;
END  IF ;
WHILE  V_TEST=1 LOOP
BEGIN
XXXX
END ;
END  LOOP ;
V_TEST := 123;
6. ⽤for in 使⽤cursor
IS
CURSOR cur IS SELECT * FROM xxx;
BEGIN
oracle游标的使用
FOR cur_result in cur LOOP
BEGIN
V_SUM :=cur_result.列名1+cur_result.列名2
END;
END LOOP;
END;
7. 带参数的cursor
CURSOR C_USER(C_ID NUMBER) IS SELECT NAME FROM USER WHERE TYPEID=C_ID;
OPEN C_USER(变量值);
LOOP
FETCH C_USER INTO V_NAME;
EXIT FETCH C_USER%NOTFOUND;
do something
END LOOP;
CLOSE C_USER;
8. ⽤pl/sql developer debug
连接数据库后建⽴⼀个Test WINDOW,在窗⼝输⼊调⽤SP的代码,F9开始debug,CTRL+N单步调试
关于oracle存储过程的若⼲问题备忘
1.在oracle中,数据表别名不能加as,如:
select a.appname from appinfo a;-- 正确
select a.appname from appinfo as a;-- 错误
也许,是怕和oracle中的存储过程中的关键字as冲突的问题吧
2.在存储过程中,select某⼀字段时,后⾯必须紧跟into,如果select整个记录,利⽤游标的话就另当别论了。
select af.keynode into kn from APPFOUNDATION af
where af.appid=aid and af.foundationid=fid;-- 有into,正确编译
select af.keynode from APPFOUNDATION af
where af.appid=aid and af.foundationid=fid;-- 没有into,编译报错,提⽰:Compilation
Error: PLS-00428: an INTO clause is expected in this SELECT statement
3.在利⽤select…into…语法时,必须先确保数据库中有该条记录,否则会报出”no data found”异常。
可以在该语法之前,先利⽤select count(*) from 查看数据库中是否存在该记录,如果存在,再利⽤select…into…
4.在存储过程中,别名不能和字段名称相同,否则虽然编译可以通过,但在运⾏阶段会报错
--正确
select keynode into kn from APPFOUNDATION where appid=aid and foundationid=fid;
--错误
select af.keynode into kn from APPFOUNDATION af
where af.appid=appid and af.foundationid=foundationid;
-- 运⾏阶段报错,提⽰ORA-01422:exact fetch returns more than requested number of rows
5.在存储过程中,关于出现null的问题
假设有⼀个表A,定义如下:
create table A(
id varchar2(50) primary key not null,
vcount number(8) not null,
bid varchar2(50) not null -- 外键
)
;
如果在存储过程中,使⽤如下语句:
select sum(vcount) into fcount from A where bid='xxxxxx';
如果A表中不存在bid=”xxxxxx”的记录,则fcount=null(即使fcount定义时设置了默认值,如:fcount number(8):=0依然⽆效,fcount还是会变成null),这样以后使⽤fcount时就可能有问题,所以在这⾥最好先判断⼀下:
if fcount is null then
fcount:=0;
end if;
这样就⼀切ok了。
6.Hibernate调⽤oracle存储过程
HibernateTemplate().execute(
new HibernateCallback() {
public Object doInHibernate(Session session)
throws HibernateException, SQLException {
CallableStatement cs = session
.connection()
.prepareCall("{call modifyapppnumber_remain(?)}");
cs.setString(1, foundationid);
return null;
}
});