ORACLE存储过程详解

ORACLE存储过程详解

1、定义

所谓存储过程(Stored Procedure),就是一组用于完成特定数据库功能的SQL语句集,该SQL语句集经过编译后存储在数据库系统中。在使用时候,用户通过指定已经定义的存储过程名字并给出相应的存储过程参数来调用并执行它,从而完成一个或一系列的数据库操作。

2、存储过程的创建

Oracle存储过程包含三部分:过程声明,执行过程部分,存储过程异常。

(1)无参存储过程语法


1

2

3

4

5

6

7

8

create or replace procedure NoParPro 

 as  //声明 

 

 begin // 执行 

 

 exception//存储过程异常 

 

 end;

(2)带参存储过程实例


1

2

3

4

5

6

7

8

9

create or replace procedure queryempname(sfindno emp.empno%type)  

as

   sName emp.ename%type; 

   sjob emp.job%type; 

begin

       .... 

exception 

       .... 

end;

(3)带参数存储过程含赋值方式


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

create or replace procedure runbyparmeters   

    (isal in emp.sal%type,  

     sname out varchar

     sjob in out varchar

 as

    icount number; 

 begin

      select count(*) into icount from emp where sal>isal and job=sjob; 

      if icount=1 then

        .... 

      else

       .... 

     end if; 

exception 

     when too_many_rows then

     DBMS_OUTPUT.PUT_LINE(‘返回值多于1行‘); 

     when others then

     DBMS_OUTPUT.PUT_LINE(‘在RUNBYPARMETERS过程中出错!‘); 

end;

其中参数IN表示输入参数,是参数的默认模式。

OUT表示返回值参数,类型可以使用任意Oracle中的合法类型。

OUT模式定义的参数只能在过程体内部赋值,表示该参数可以将某个值传递回调用他的过程

IN OUT表示该参数可以向该过程中传递值,也可以将某个值传出去。

(4)存储过程中游标定义使用


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

as //定义(游标一个可以遍历的结果集)  

CURSOR cur_1 IS

  SELECT area_code,CMCODE,SUM(rmb_amt)/10000 rmb_amt_sn, 

         SUM(usd_amt)/10000 usd_amt_sn  

  FROM BGD_AREA_CM_M_BASE_T  

  WHERE ym >= vs_ym_sn_beg  

       AND ym <= vs_ym_sn_end  

  GROUP BY area_code,CMCODE;  

begin //执行(常用For语句遍历游标)      

FOR rec IN cur_1 LOOP  

  UPDATE xxxxxxxxxxx_T  

   SET rmb_amt_sn = rec.rmb_amt_sn,usd_amt_sn = rec.usd_amt_sn  

   WHERE area_code = rec.area_code  

   AND CMCODE = rec.CMCODE  

   AND ym = is_ym;  

END LOOP;

(5)游标的定义


1

2

3

4

5

6

7

8

9

10

11

12

--显示cursor的处理

declare

---声明cursor,创建和命名一个sql工作区

cursor cursor_name is

    select real_name from account_hcz;

    v_realname varchar2(20);

begin

    open cursor_name;---打开cursor,执行sql语句产生的结果集

    fetch cursor_name into v_realname;--提取cursor,提取结果集中的记录

    dbms_output.put_line(v_realname);

    close cursor_name;--关闭cursor

end;

3、在Oracle中对存储过程的调用

(1)过程调用方式一


1

2

3

4

5

6

7

8

9

10

11

declare

      realsal emp.sal%type; 

      realname varchar(40); 

      realjob varchar(40); 

begin   //过程调用开始 

      realsal:=1100; 

      realname:=‘‘

      realjob:=‘CLERK‘

      runbyparmeters(realsal,realname,realjob);--必须按顺序 

      DBMS_OUTPUT.PUT_LINE(REALNAME||‘   ‘||REALJOB); 

END;  //过程调用结束

(2)过程调用方式二


1

2

3

4

5

6

7

8

9

10

11

12

declare

     realsal emp.sal%type; 

     realname varchar(40); 

     realjob varchar(40); 

begin    //过程调用开始 

     realsal:=1100; 

     realname:=‘‘

     realjob:=‘CLERK‘

     --指定值对应变量顺序可变 

     runbyparmeters(sname=>realname,isal=>realsal,sjob=>realjob);          

    DBMS_OUTPUT.PUT_LINE(REALNAME||‘   ‘||REALJOB); 

END;  //过程调用结束

(3)过程调用方式三(SQL命令行方式下)

1、SQL>exec proc_emp(‘参数1’,’参数2’);//无返回值过程调用

2、SQL>var vsal number

SQL> exec proc_emp (‘参数1’,:vsal);// 有返回值过程调用

或者:call proc_emp (‘参数1’,:vsal);// 有返回值过程调用

存储过程创建语法


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

create [or replace] procedure 存储过程名(param1 in type,param2 out type)

as

变量1 类型(值范围);

变量2 类型(值范围);

Begin

    Select count(*) into 变量1 from 表A where列名=param1;

    If (判断条件) then

       Select 列名 into 变量2 from 表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;

注意事项

存储过程参数不带取值范围,in表示传入,out表示输出; 变量带取值范围,后面接分号; 在判断语句前最好先用count(*)函数判断是否存在该条操作记录; 用select … into … 给变量赋值; 在代码中抛异常用 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. 基本结构


1

2

3

4

5

6

7

8

9

10

CREATE OR REPLACE PROCEDURE 存储过程名字

(

    参数1 IN NUMBER,

    参数2 IN NUMBER

) IS

变量1 INTEGER :=0;

变量2 DATE;

BEGIN

    --执行体

END 存储过程名字;

2. SELECT INTO STATEMENT

将select查询的结果存入到变量中,可以同时将多个列存储多个变量中,必须有一条记录,否则抛出异常(如果没有记录抛出NO_DATA_FOUND)

例子:


1

2

3

4

5

6

BEGIN

SELECT col1,col2 into 变量1,变量2 FROM typestruct where xxx;

EXCEPTION

WHEN NO_DATA_FOUND THEN

    xxxx;

END;

3. IF 判断


1

2

3

4

5

IF V_TEST = 1 THEN

   BEGIN

      do something

   END;

 END IF;

4. while 循环


1

2

3

4

5

WHILE V_TEST=1 LOOP

BEGIN

  XXXX

END;

END LOOP;

5. 变量赋值


1

V_TEST := 123;

6. 用for in 使用cursor


1

2

3

4

5

6

7

8

9

IS

 CURSOR cur IS SELECT * FROM xxx;

 BEGIN

FOR cur_result in cur LOOP

 BEGIN

  V_SUM :=cur_result.列名1+cur_result.列名2

 END;

END LOOP;

 END;

7. 带参数的cursor


1

2

3

4

5

  CURSOR C_USER(C_ID NUMBER) IS SELECT NAME FROM USER WHERE TYPEID=C_ID;

  OPEN C_USER(变量值);

FETCH C_USER INTO V_NAME;

  EXIT WHEN FETCH C_USER%NOTFOUND;

CLOSE C_USER;

8. 用pl/sql developer debug

连接数据库后建立一个Test WINDOW,在窗口输入调用SP的代码,F9开始debug,CTRL+N单步调试

关于oracle存储过程的若干问题备忘

1.在oracle中,数据表别名不能加as,如:


1

2

select a.appname from appinfo a;-- 正确

select a.appname from appinfo as a;-- 错误

也许,是怕和oracle中的存储过程中的关键字as冲突的问题吧

2.在存储过程中,select某一字段时,后面必须紧跟into,如果select整个记录,利用游标的话就另当别论了。


1

2

3

4

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.在存储过程中,别名不能和字段名称相同,否则虽然编译可以通过,但在运行阶段会报错


1

2

3

4

5

6

--正确

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,定义如下:


1

2

3

4

5

create table A(

id varchar2(50) primary key not null,

vcount number(8) not null,

bid varchar2(50) not null -- 外键

);

如果在存储过程中,使用如下语句:


1

select sum(vcount) into fcount from A where bid=‘xxxxxx‘;

如果A表中不存在bid=”xxxxxx”的记录,则fcount=null(即使fcount定义时设置了默认值,如:fcount number(8):=0依然无效,fcount还是会变成null),这样以后使用fcount时就可能有问题,所以在这里最好先判断一下:


1

2

3

if fcount is null then

    fcount:=0;

end if;

这样就一切ok了。

6.Hibernate调用oracle存储过程


1

2

3

4

5

6

7

8

9

10

11

12

this.pnumberManager.getHibernateTemplate().execute(

            new HibernateCallback() {

                public Object doInHibernate(Session session)

                        throws HibernateException, SQLException {

                    CallableStatement cs = session

                            .connection()

                            .prepareCall("{call modifyapppnumber_remain()}");

                    cs.setString(1, foundationid);

                    cs.execute();

                    return null;

                }

            });

转载自:https://www.2cto.com/database/201802/723493.html

原文地址:https://www.cnblogs.com/guohu/p/11007350.html

时间: 2024-08-27 00:52:44

ORACLE存储过程详解的相关文章

Oracle存储过程详解(引用)+补充(转)

一.过程 (存储过程) 过程是一个能执行某个特定操作的子程序.使用CREATE OR REPLACE创建或者替换保存在数据库中的一个子程序.示例1:声明存储过程,该过程返回dept表行数 DECLARE PROCEDURE getDeptCount AS deptCount INT; BEGIN SELECT COUNT(*) INTO deptCount FROM DEPT; DBMS_OUTPUT.PUT_LINE('DEPT表的共有记录数:'||deptCount); END getDep

问题:Oracle出发器;结果:1、Oracle触发器详解,2、Oracle触发器示例

ORACLE触发器详解 本篇主要内容如下: 8.1 触发器类型 8.1.1 DML触发器 8.1.2 替代触发器 8.1.3 系统触发器 8.2 创建触发器 8.2.1 触发器触发次序 8.2.2 创建DML触发器 8.2.3 创建替代(INSTEAD OF)触发器 8.2.3 创建系统事件触发器 8.2.4 系统触发器事件属性 8.2.5 使用触发器谓词 8.2.6 重新编译触发器 8.3 删除和使能触发器 8.4 触发器和数据字典 8.5   数据库触发器的应用举例 触发器是许多关系数据库系

oracle权限详解

一.权限分类:系统权限:系统规定用户使用数据库的权限.(系统权限是对用户而言). 实体权限:某种权限用户对其它用户的表或视图的存取权限.(是针对表或视图而言的). 二.系统权限管理:1.系统权限分类:DBA: 拥有全部特权,是系统最高权限,只有DBA才可以创建数据库结构. RESOURCE:拥有Resource权限的用户只可以创建实体,不可以创建数据库结构. CONNECT:拥有Connect权限的用户只可以登录Oracle,不可以创建实体,不可以创建数据库结构. 对于普通用户:授予connec

MySQL存储过程详解 mysql 存储过程

MySQL存储过程详解  mysql 存储过程 (2010-12-13 10:59:38) 转载▼ 标签: it 分类: mysql mysql存储过程详解 1.      存储过程简介   我们常用的操作数据库语言SQL语句在执行的时候需要要先编译,然后执行,而存储过程(Stored Procedure)是一组为了完成特定功能的SQL语句集,经编译后存储在数据库中,用户通过指定存储过程的名字并给定参数(如果该存储过程带有参数)来调用执行它. 一个存储过程是一个可编程的函数,它在数据库中创建并保

MySQL存储过程详解 mysql 存储过程(转:http://blog.sina.com.cn/s/blog_52d20fbf0100ofd5.html)

转:http://blog.sina.com.cn/s/blog_52d20fbf0100ofd5.html mysql存储过程详解 1.      存储过程简介   我们常用的操作数据库语言SQL语句在执行的时候需要要先编译,然后执行,而存储过程(Stored Procedure)是一组为了完成特定功能的SQL语句集,经编译后存储在数据库中,用户通过指定存储过程的名字并给定参数(如果该存储过程带有参数)来调用执行它. 一个存储过程是一个可编程的函数,它在数据库中创建并保存.它可以有SQL语句和

oracle 序列 详解

序列: 是oacle提供的用于产生一系列唯一数字的数据库对象. l  自动提供唯一的数值 l  共享对象 l  主要用于提供主键值 l  将序列值装入内存可以提高访问效率 创建序列: 1.  要有创建序列的权限 create sequence 或 create any sequence 2.  创建序列的语法 CREATE SEQUENCE sequence  //创建序列名称        [INCREMENT BY n]  //递增的序列值是n 如果n是正数就递增,如果是负数就递减 默认是1

Oracle SGA详解

SGA(SYSTEM Global Area )系统全局区 l 数据高速缓存 在Oracle进行数据处理的过程中,代价最昂贵的就是物理 I/O操作了.同样的数据从内存中得到要比从磁盘上读取快的多.因此,优化Oracle的一个重要的目标就是尽可能的降低物理 I/O操作. Oracle的 Buffer Cache用于缓存从磁盘中读取的数据,当 Oracle需要查找某些信息的时候,首先会在 BufferCache中寻找,如果找到了,则直接将结果返回.如果找不到,则需要对磁盘进行扫描, Oracle将在

Oracle ROWID详解

1.ROWID定义 ROWID:数据库中行的全局唯一地址 对于数据中的每一行,rowid伪列返回行的地址.rowid值主要包含以下信息: 对象的数据对象编号 该行所在的数据文件中的数据块 该行中数据块的位置(第一行是0) 数据行所在的数据文件(第一个文件是1).该文件编号是相对于表空间. 通常来说,一个rowid值唯一标识数据中的一行.然而,存储在同一聚簇中不同的表可以有相同的rowid. 2.扩展ROWID 从Oracle 8i开始使用扩展rowid标识行物理地址 扩展rowid使用base6

.net调用存储过程详解

连接字符串 string conn = ConfigurationManager.ConnectionStrings["NorthwindConnectionString"].ConnectionString; confige文件 <connectionStrings>         <add name="NorthwindConnectionString" connectionString="Data Source=.;Initial