文章分类:数据库存储过程创建语法:(1)无参create or replace procedure 存储过程名as变量1类型(值范围);变量2类型(值范围);BeginExcepti onEnd;(2)带参create or replace procedure 存储过程名(param1 in type,param2 out type)as变量1类型(值范围);变量2类型(值范围);BeginSelect count(*) into 变量1 from 表A where 列名=param1 ;If (判断条件)thenSelect 列名into 变量2 from 表A where 列名=param1;Dbms_output.Put_line(打印信息');Elseif (判断条件)thenDbms_output.Put_line(打印信息');ElseRaise 异常名(NO DATA FOUND);End if;Excepti onWhen others the nRollback;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 除数为0SUBSCRIPT_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 create or replace procedure run byparmeters (isal in emp.sal%type,sn ame out varchar,sjob in out varchar)2 as ico unt nu mber;3 begin4 select coun t(*) into ico unt from emp where sal>isal and job=sjob;5 if ico un t=1 the n6 ....9 else10 ....12 end i f;13 exceptio n14 whe n too_many_rows the n15 DBMS OUTPUT.PUT LINE('返回值多于1 行');16 whe n others the n17 DBMS OUTPUT.PUT LINE('在RUNBYPARMETERS 过程中出错!');18 en d;过程调用方式一1 declare2 realsal emp.sal%type;3 realn ame varchar(40);4 realjob varchar(40);5 begin6 realsal:=1100;7 realn ame:=";8 realjob:='CLERK:9 run byparmeters(realsal,rea In ame,realjob); ---------- 必须按顺序10 DBMS_OUTPUT.PUT_LINE(REALNAME『’||REALJOB);11 END;12方式二1 declare2 realsal emp.sal%type;3 realn ame varchar(40);4 realjob varchar(40);5 begin6 realsal:=1100;7 realn ame:=";8 realjob:='CLERK';9 run byparmeters(s name=>rea In ame,isal=>realsal,sjob=>realjob); ------ 指定值对应变量顺序可变10 DBMS OUTPUT.PUT LINE(REALNAME『'||REALJOB);11 END;说明:(1)使用%TYPE在许多情况下,PL/SQL变量可以用来存储在数据库表中的数据。
在这种情况下,变量应该拥有与表列相同的类型。
例如,students表的first_name列的类型为VARCHAR2(20),我们可以按照下述方式声明一个变量DECLAREv_FirstName VARCHAR2(20);但是如果first_name列的定义改变了会发生什么(比如说表改变了,first_name现在的类型变为VARCHAR2(25) )?那就会导致所有使用这个列的PL/SQL代码都必须进行修改。
如果你有很多的PL/SQL代码,这种处理可能是十分耗时和容易出错的。
这时,你可以使用” %TYP”属性而不是将变量类型硬性编码。
(2)使用%ROWTYPE (相当于定义一个struct来进行存放,以对象来看对数据)在PL/SQL中将一个记录声明为具有相同类型的数据库行的作法是很常见的。
PL/SQL提供了%ROWTYPE运算符,使得这样的操作更为方便。
例如:DECLAREv_Stude ntRecord stude nts%ROWTYPE;将定义一个记录,该记录中的字段将与stude nts表中的列相对应。
例如:declarev_jobs hr.jobs%rowtype;beginselect*in tov_jobsfromhr.jobswhere job_id ='&aa:dbms_output.put_line('序号'||v_jobs.job_id ); dbms_output.put_line('名称'||v_jobs.job_title); en d;执行,我们输入aa变量的值:AD_VP 输出结果为:序号AD_VP 名称Admi nistratio n Vice Preside nt--初始化””” ”” ””””””” ”” …vaw …” ” r.w ” ” JVW ”” .m ” ” -m ” ww ” ”” ” ■try ・” fare."”””””” ” .vf5” …v'fc ””””ZZ”- JVW ”””” -m ” ■ ww ” ww ” ” aw^ ” ”” ” 02 ” ”” ” ”””””” ”.w ” ””””” ” Z2 ” …” ”以下是代码片段:select max(statid) into n ewID from sh_gprsstreamstat;if (newID is null) thennewID := 1;end if;for v_SPINFO In c_SPINFO loop -- 首先获取SPID--其次遍历出与当前SPID对应的所有MDT以下是代码片段:!open c MDTINFo for select distinct▼ —mdtid from sh_mdt info where (isactive = '0')and (spid = v_SPINFO.spid);loopfetch c_MDTINFO into v_MDTINFO;!exit whe n c_MDTINFO% notfou nd;i iv_UpC onten tLe ns := 0; _ ___ _ ___________________ _______v_UpTra nsConten tLe ns := 0; v_UpTotalLe ns := 0; v_Dow nConten tLe ns := 0; v_Dow nTotalLe ns := 0;(ST_NUMIN NUMBER, ED_NUMINNUMBERdeclarei nu mber;beginFOR i IN ST_NUM..ED_NUM LOOP INSERT INTO tb values(i,i,3,3,3,1OO,'O'); END LOOP; en d; END; 运行:sql>executeINSERTAMOUNTTEST(1,45OOO) 2、从存储过程中返回值create or replace procedure spaddflowdate --一次插入45000条测试数据( varAppTypeId varFlowId DateLe ngth ReturnValue in varchar2, in varchar2, in nu mber, out nu mber--返回值)is beginin sert into td values(varAppTypeld,varFlowld,DateLe ngth) retur ning 1 into Retur nV alue; --返回值 commit; exception when others the n rollback; en d;存储过程的执行sql>variable testvalue nu mber; 1、用来插入大量测试数据的存储过程CREATE OR REPLACE PROCEDURE INSERTAMOUNTTEST ) IS BEGINsql>execute spaddflowdate('v','v',2,:testvalue); sql>pri nt就可以看到执行结果3、用包实现存储过程返回游标: create or replace package test_p as type outList is ref cursor; PROCEDURE geti nfor(taxpayerList end test_p;/ create or replace package body get in for(taxpayerList out outList) isOPEN taxpayerList FORtd where tag='0';end geti nfor; end test_p; /运行:set serverout on; --将输出工具打开variable x refcursor; execute test_p.geti nfor(:x); exec test_p.get in for(:x); print x;drop package test_p;oracle 存储过程的基本语法1. 基本结构CREATE OR REPLACE PROCEDURE 存储过程名字(参数 1 IN NUMBER,out outList);test_p as PROCEDURE begin select * from参数 2 IN NUMBER)IS变量 1 INTEGER :=0;变量2 DATE;BEGINEND存储过程名字2.SELECT INTO STATEMENT将select查询的结果存入到变量中,可以同时将多个列存储多个变量中,必须有一条记录,否则抛岀异常(如果没有记录抛岀NO_DATA_FOUND)例子:BEGINSELECT col1,col2 into 变量1,变量2 FROM typestruct where xxx;EXCEPTIONWHEN NO_DATA_FOUND THENxxxx;END;3.IF 判断IF V_TEST=1 THENBEGINdo someth ingEND;END IF;4. while 循环WHILE V_TEST=1 LOOPBEGINXXXXEND;END LOOP;5. 变量赋值V_TEST := 123;6. 用for in 使用cursorISCURSOR cur IS SELECT * FROM xxx;BEGINFOR cur_result in cur LOOPBEGINV_SUM :=cur_result. 列名1+cur_result. 列名2END;END LOOP;END;7. 带参数的cursorCURSOR C_USER(C_ID NUMBER) IS SELECT NAME FROM USER WHERE TYPEID=C_ID;OPEN C_USER(变量值);LOOPFETCH C_USER INTO V_NAME;EXIT FETCH C_USER%NOTFOUND;do someth ingEND 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整个记录,利用游标的话就另当别论了。