oracle 存储过程

来源:互联网 发布:mysql使用 编辑:程序博客网 时间:2024/05/29 10:34

1. oracle 存储过程基本语法

create or replace procedure 存储过程名(param1 in type,param2 out type) as 变量1 类型(值范围); --vs_msg   VARCHAR2(4000); 变量2 类型(值范围);CURSOR c1 IS  SELECT  * FROMdat_trade;Begin         业务逻辑;EXCEPTION    WHEN OTHERS THEN           异常处理;   COMMIT;    RETURN;End;

ps:

a. 存储过程参数不带取值范围,in表示传入,out表示输出类型,,可以使用任意Oracle中的合法类型,,也可以使用%type 来获取参数的类型(table_name.column_name%TYPE);

b.变量带取值范围,后面接分号;也可以定义游标(简单的说就是一个可以遍历的结果集) ;


2. 常用处理方法

a.条件判断:if ... elsif ... else ... 判断;case ... when ... end case 判断;

example:

CREATEORREPLACEPROCEDUREproc_test(v_numINNUMBER)ASBEGIN   IF v_num<10   THEN      DBMS_OUTPUT.put_line(v_num);   ELSIF v_num>10AND v_num<50   THEN      DBMS_OUTPUT.put_line(v_num -10);   ELSE      DBMS_OUTPUT.put_line(v_num -50);   ENDIF;END proc_test;

b.循环:for ... in ... loop循环; loop循环; while循环

example:

CREATEORREPLACEPROCEDURE proc_testAS   CURSOR c1   IS      SELECT  * FROMdat_trade;BEGIN   FOR xIN c1   LOOP      DBMS_OUTPUT.put_line(x.id);   ENDLOOP;END proc_test;

c.数组

首先明确一个概念:Oracle 中本是没有数组的概念的,数组其实就是一张表(Table), 每个数组元素就是表中的一个记录。 

使用数组时,用户可以使用Oracle 已经定义好的数组类型,或可根据自己的需要定义数组类型。 

(1) 使用Oracle 自带的数组类型 

x array; -- 使用时需要需要进行初始化 

e.g: 

create or replace procedure test(y out array) is  x array;    begin x := new array(); y := x; end test; 


(2) 自定义的数组类型 ( 自定义数据类型时,建议通过创建Package 的方式实现,以便于管理

create or replace package myPackage is    Public type declarations   type info is record(     name varchar(20),     y number);   type TestArray is table of info index by binary_integer;   

-- 此处声明了一个TestArray 的类型数据,其实其为一张存储Info 数据类型的Table 而已,及TestArray 就是一张表,有两个字段,一个是name ,一个是。需要注意的是此处使用了Index by binary_integer 编制该Table 的索引项,也可以不写,直接写成:type TestArray is 

table of info ,如果不写的话使用数组时就需要进行初始化:

varArray myPackage.TestArray; varArray := new myPackage.TestArray(); end TestArray; 


d.游标

(1)Cursor 型游标不能用于参数传递)

create or replace procedure test() is   Cursor cusor_1 is select std_name from student where  ...;  --Cursor 的使用方式1   cursor_2 Cursor; begin select class_name into cursor_2 from class where ...;  --Cursor 的使用方式2 

可使用For x in cursor LOOP .... end LOOP; 来实现对Cursor 的遍历

end test;

(2)SYS_REFCURSOR 型游标,该游标是Oracle 以预先定义的游标,可作出参数进行传递


create or replace procedure test(rsCursor out SYS_REFCURSOR) is cursor SYS_REFCURSOR; name varhcar(20); begin OPEN cursor FOR select name from student where ... --SYS_REFCURSOR 只能通过OPEN 方法来打开和赋值 LOOP fetch cursor into name   --SYS_REFCURSOR 只能通过fetch into 来打开和遍历 exit when cursor%NOTFOUND;              --SYS_REFCURSOR 中可使用三个状态属性:                                                   ---%NOTFOUND( 未找到记录信息) %FOUND( 找到记录信息)                                                                                 ---%ROWCOUNT( 然后当前游标所指向的行位置)  dbms_output.putline(name); end LOOP; rsCursor := cursor; end test; 

3.综合示例 现假设存在两张表,

一张是学生成绩表(studnet) ,字段为:stdId,math,article,language,music,sport,total,average,step              

一张是学生课外成绩表(out_school), 字段为:stdId,parctice,comment 通过存储过程自动计算出每位学生的总成绩和平均成绩,

同时,如果学生在课外课程中获得的评价为A ,就在总成绩上加20 分。

create or replace procedure autocomputer(step in number) is rsCursor SYS_REFCURSOR; commentArray myPackage.myArray; math number; article number; language number; music number; sport number; total number; average number; stdId varchar(30); record myPackage.stdInfo; i number; begin i := 1; get_comment(commentArray); -- 调用名为get_comment() 的存储过程获取学生课外评分信息 OPEN rsCursor for select stdId,math,article,language,music,sport from student t where t.step = step; LOOP fetch rsCursor into stdId,math,article,language,music,sport; exit when rsCursor%NOTFOUND; total := math + article + language + music + sport; for i in 1..commentArray.count LOOP   record := commentArray(i);     if stdId = record.stdId then    begin       if record.comment = 'A' then        begin           total := total + 20;       go to next; -- 使用go to 跳出for 循环          end;     end if;   end;   end if; end LOOP; <<continue>>  average := total / 5;  update student t set t.total=total and t.average = average where t.stdId = stdId; end LOOP; end; end autocomputer; -- 取得学生评论信息的存储过程 create or replace procedure get_comment(commentArray out myPackage.myArray) is rs SYS_REFCURSOR ; record myPackage.stdInfo; stdId varchar(30); comment varchar(1); i number; begin open rs for select stdId,comment from out_school i := 1; LOOP  fetch rs into stdId,comment; exit when rs%NOTFOUND; record.stdId := stdId;  record.comment := comment; recommentArray(i) := record; i:=i + 1; end LOOP; end get_comment; -- 定义数组类型myArray create or replace package myPackage is begin type stdInfo is record(stdId varchar(30),comment varchar(1)); type myArray is table of stdInfo index by binary_integer; end myPackage;


4.oracle存储过程的调试

http://soft.chinabyte.com/database/332/12270332.shtml

ps.如果单步跟踪不能进入存储过程源码,右键点击存储过程->Recompile.


参考

http://www.cnblogs.com/hero4china/articles/base_rule_oracle_procedure.html




0 0
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 睾丸被压了好痛怎么办 睾丸皮肤痒破了怎么办 鸡儿下面了蛋痒怎么办? 射精后小腹胀该怎么办 手压伤了有淤血怎么办 手挤压伤了肿了怎么办 手被挤压肿了怎么办 手砸伤了肿了怎么办 手被机器压伤了怎么办 上眼皮眼睛肿了怎么办 上眼皮内有淤血怎么办 种睫毛眼睛红痛怎么办 一只眼睛变红了怎么办 黑眼球缺了一角怎么办 眼镜度数配高了怎么办 孩子近视800度可怎么办 儿童眼睛近视怎么办才能恢复正常 小孩眼睛近视怎么办才能恢复正常 3岁宝宝近视怎么办啊 6个月婴儿近视怎么办 近视眼的人老了怎么办 一千多度的近视怎么办 近视镜片磨花了怎么办 眼镜镜片磨花了怎么办 戴眼镜鼻梁有印怎么办 狗狗发烧怎么办最有效 狗狗感冒怎么办最有效 狗狗发烧去医院怎么办 眼睛里长了虫子怎么办 吃了没熟的猪肉怎么办 没熟的鸡肉吃了怎么办 狗狗大便有绦虫怎么办 吃了有虫的猪肉怎么办 米猪肉吃了会怎么办 吃了鱼的寄生虫怎么办 鱼身体里有线虫怎么办 幼猫半夜一直叫怎么办 猫半夜4点叫不停怎么办 眼睛里长了黄斑怎么办 眼睛周围长小疙瘩怎么办 眼睛被手机砸了怎么办