Oracle中procedure/cursor深入分析

procedure概述

存储过程( Stored Procedure )是一组为了完成特定功能的 SQL 语句集,经编译后存储在数据库中。

用户通过指定存储过程的名字并给出参数(如果该存储过程带有参数)来执行它。

存储过程是由流控制和 SQL 语句书写的过程,这个过程经编译和优化后存储在数据库服务器中,应用程序使用时只要调用即可。

在Oracle 中,若干个有联系的过程可以组合在一起构成程序包。

procedure优点

存储过程只在创造时进行编译,以后每次执行存储过程都不需再重新编译,而一般 SQL 语句每执行一次就编译一次,所以使用存储过程可提高数据库执行速度。

当对数据库进行复杂操作时(如对多个表进行 Update、Insert、Query、Delete时),可将此复杂操作用存储过程封装起来与数据库提供的事务处理结合一起使用。

存储过程可以重复使用,可减少数据库开发人员的工作量。

安全性高,可设定只有某用户才具有对指定存储过程的使用权。

和function的区别

procedure例子

CREATE OR REPLACE procedure proc_trade(
       v_tradeid     in number, --交易id
       v_third_ip    in varchar2, --第三方ip
       v_third_time  in date, --第三方完成时间
       v_thire_state in number, --第三方状态
       o_result      out number, --返回值
       o_detail      out varchar2 --详细描述
      ) as
  -- 定义变量
  v_error varchar2(500);
begin
  --对变量赋值
  o_result := 0;
  o_detail := '验证失败';
  --业务逻辑处理
  if v_tradeid > 100 then
    insert into table_name (.. .) values (.. .);
    commit;
  elsif v_tradeid < 100 and v_tradeid > 50 then
    insert into table_name (.. .) values (.. .);
    commit;
  else
    goto log;
  end if;
  --跳转标志符,名称自己指定
  <<log>>
  o_result := 1;
  --捕获异常
exception
  when no_data_found then
    result := 2;
  when dup_val_on_index then
    result := 3;
  when others then
    result := -1;
end proc_trade;
参数类型可以自己指定,这种写法可行,但是最好使用%type 来获取参数的类型(table_name.column_name%TYPE)。 这样就不会出现参数类型的错误。

存储过程中的循环

for … in … loop 循环

循环遍历游标

示例1:

CREATE OR REPLACE PROCEDURE proc_test AS
  CURSOR c1 IS
    SELECT * FROM dat_trade;
BEGIN
  FOR x IN c1 LOOP
    DBMS_OUTPUT.put_line(x.id);
  END LOOP;
END proc_test;
示例 2:

CREATE OR REPLACE PROCEDURE proc_test AS
BEGIN
  FOR x IN (SELECT power_id FROM sys_power) LOOP
    DBMS_OUTPUT.put_line(x.power_id);
  END LOOP;
END proc_test;
根据数值进行循环

栗子一

CREATE OR REPLACE PROCEDURE proc_test() AS
BEGIN
  for x in 1 .. 100 loop
    dbms_output.put_line(x);
  end loop;
END proc_test;
栗子2:在过程里指定输入参数 v_num. 在调用过程时指定循环次数。

CREATE OR REPLACE PROCEDURE proc_test(v_num IN NUMBER) AS
BEGIN
  FOR x IN 1 .. v_num LOOP
    DBMS_OUTPUT.put_line(x);
  END LOOP;
END proc_test;
loop 循环

LOOP
DELETE FROM orders
WHERE senddate < TO_CHAR (ADD_MONTHS (SYSDATE, -3),
'yyyy-mm-dd')
AND ROWNUM < 1000;
EXIT WHEN SQL%ROWCOUNT < 1;
COMMIT;
END LOOP;
这 里 的 SQL%ROWCOUNT 是 隐 士 游 标 。 除 了 这 个 , 还 有 其 他 几
个: %found, %notfound, %isopen。

while 循环

CREATE OR REPLACE PROCEDURE proc_test(v_num IN NUMBER)
  AS
  i NUMBER := 1;
BEGIN
  WHILE i < v_num LOOP
    BEGIN
      i := i + 1;
      DBMS_OUTPUT.put_line(i);
    END;
  END LOOP;
END proc_test;
存储过程中的判断

if … elsif … else … 判断

CREATE OR REPLACE PROCEDURE proc_test(v_num IN NUMBER) AS
BEGIN
  IF v_num < 10 THEN
    DBMS_OUTPUT.put_line(v_num);
  ELSIF v_num > 10 AND v_num < 50 THEN
    DBMS_OUTPUT.put_line(v_num - 10);
  ELSE
    DBMS_OUTPUT.put_line(v_num - 50);
  END IF;
END proc_test;
case … when … end case 判断

CREATE OR REPLACE PROCEDURE proc_test(v_num IN NUMBER) AS
BEGIN
  case v_num
    when 1 then
      DBMS_OUTPUT.put_line(v_num);
    when 2 then
      DBMS_OUTPUT.put_line(v_num);
    when 3 then
      DBMS_OUTPUT.put_line(v_num);
    else
      null;
  end case;
END proc_test;
游标

之前整理的游标的知识

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

CREATE OR REPLACE PROCEDURE proc_test AS
  CURSOR c1 IS
    SELECT * FROM dat_trade;
BEGIN
  FOR x IN c1 LOOP
    DBMS_OUTPUT.put_line(x.id);
  END LOOP;
END proc_test;
SYS_REFCURSOR 型游标

该游标是 Oracle 预先定义的游标,可作出参数进行传递。

SYS_REFCURSOR 只能通过 OPEN 方法来打开和赋值

我们可以使用这种类似的游标来返回一个结果集:

CREATE OR REPLACE procedure proc_test(
    checknum   in number, --每次返回的数据量
    ref_cursor out sys_refcursor --返回的结果集,游标
    ) as
begin
  open ref_cursor for
    select *
      from (select * from dat_trade where state = 41 order by id)
     where rownum < checknum;
end proc_test;
SYS_REFCURSOR 中可使用四个状态属性:

( 1) . %NOTFOUND(未找到记录信息)
( 2) . %FOUND(找到记录信息)
( 3) . %ROWCOUNT(然后当前游标所指向的行位置)
(4). %ISOPEN(是否打开)
CREATE OR REPLACE PROCEDURE proc_test(
    checknum   IN NUMBER, --每次返回的数据量
    ref_cursor OUT sys_refcursor --返回的结果集,游标
    ) AS
  t_tmp table_name%ROWTYPE;
BEGIN
  OPEN ref_cursor FOR
    SELECT *
      FROM (SELECT * FROM table_name WHERE state = 41 ORDER BY id)
     WHERE ROWNUM < checknum;
  --循环游标
  LOOP
    FETCH ref_cursor
      INTO t_tmp;
    EXIT WHEN ref_cursor%NOTFOUND;
    -- DBMS_OUTPUT.put_line (t_tmp.id);
    UPDATE table_name SET state = 53 WHERE id = t_tmp.id;
    COMMIT;
  END LOOP;
  CLOSE ref_cursor;
END proc_test;

时间: 2024-10-28 17:15:30

Oracle中procedure/cursor深入分析的相关文章

oracle中REF Cursor用法

1,什么是 REF游标 ?  动态关联结果集的临时对象.即在运行的时候动态决定执行查询.   2,REF 游标 有什么作用?  实现在程序间传递结果集的功能,利用REF CURSOR也可以实现BULK SQL,从而提高SQL性能.   3,静态游标和REF 游标的区别是什么?  ①静态游标是静态定义,REF 游标是动态关联:  ②使用REF 游标需REF 游标变量.  ③REF 游标能做为参数进行传递,而静态游标是不可能的.   4,什么是REF 游标变量?  REF游标变量是一种 引用 REF

oracle中Shared pool深入分析及性能调整

摘要:本文首先详细介绍了oracle中shared pool的概念以及所包含的内存结构.然后深入介绍了oracle对于shared pool的管理机制.最后全面介绍了有关buffer cache监控以及调优的实用方法. 1. shared pool的概念 oracle数据库作为一个管理数据的产品,必须能够认出用户所提交的管理命令(通常叫做SQL语句),从而进行响应.认出的过程叫做解析SQL语句的过程,响应的过程叫做执行SQL语句的过程.解析的过程是一个相当复杂的过程,它要考虑各种可能的异常情况,

Oracle中等待事件深入分析

概述 等待事件主要可以分为两类,即空闲(IDLE)等待事件和非空闲(NON-IDLE)等待事件. 1).空闲等待事件 ORACLE正等待某种工作,在诊断和优化数据库的时候,不用过多注意这部分事件. 2). 非空闲等待事件 专门针对 ORACLE 的活动,指数据库任务或应用运行过程中发生的等待,这些等待事件是 在调整数据库的时候需要关注与研究的. 在 Oracle 10g 中的等待事件可以通过 v$event_name 视图来查看等待事件的相关信息. 常见等待事件 Buffer busy wait

索引在Oracle中的应用深入分析_oracle

索引是提高数据查询最有效的方法,也是最难全面掌握的技术,因为正确的索引可能使效率提高10000倍,而无效的索引可能是浪费了数据库空间,甚至大大降低查询性能. 一.索引的管理成本1. 存储索引的磁盘空间2. 执行数据修改操作(INSERT.UPDATE.DELETE)产生的索引维护3. 在数据处理时所需额外的回退空间. 二.实际数据修改测试:一个表有字段A.B.C,同时进行插入10000行记录测试在没有建索引时平均完成时间是2.9秒在对A字段建索引后平均完成时间是6.7秒在对A字段和B字段建索引后

Oracle中RAISE异常深入分析_oracle

有三种方式抛出异常 1.通过PL/SQL运行时引擎 2.使用RAISE语句 3.调用RAISE_APPLICATION_ERROR存储过程 当数据库或PL/SQL在运行时发生错误时,一个异常被PL/SQL运行时引擎自动抛出.异常也可以通过RAISE语句抛出 RAISE exception_name; 显式抛出异常是程序员处理声明的异常的习惯用法,但RAISE不限于声明了的异常,它可以抛出任何任何异常.例如,你希望用TIMEOUT_ON_RESOURCE错误检测新的运行时异常处理器,你只需简单的在

oracle中游标(Cursor)的详解

概述 也许大家对数据库中游标都不陌生,但对于其概念可能有时又会有些模糊,游标到底是什么? 为了使大家对游标有个清晰的认识,本文将介绍Oracle中游标(curosr)相关的知识. 游标的概念 一般来讲,游标包含着两种不同的概念: 程序中的游标(Program Cursor)和Oracle中的游标(Oracle Curosr). 程序中的游标(Program Cursor): 在最终用户程序(Client Application)中,游标(curosr)通常指和SQL语句关联的一个数据结构,用于关

ORACLE 中定义全局变量,在其他各个PROCEDURE中使用

问题描述 ORACLE 中定义全局变量,在其他各个PROCEDURE中使用 如何在ORACLE中定义一个全局性的常量类,在其他的各个存储过程中调用使用: 类似与JAVA中得常量类这样. 我目前想到两种方案, 1,定义一个表,在这个表中录入各种变量,如何在存储过程中调用(感觉存储过程中,还是硬编码了一些东西) 2,写一个存储过程包,里面都定义function,一个function返回一个变量值(感觉一个变量写一个function太麻烦) 请问各路大神,有没有那种类似JAVA 或者C中,定义一个常量

Oracle中的临时表用法汇总

oracle|临时表 说明:下文中的一些说明和示例代码摘自CSDN,恕不一一指明出处,在此一并对相关作者表示感谢! 如果作者有异议,请来信说明  1 语法 在Oracle中,可以创建以下两种临时表: 1) 会话特有的临时表 CREATE GLOBAL TEMPORARY <TABLE_NAME> (<column specification> ) ON COMMIT PRESERVE ROWS: 2) 事务特有的临时表 CREATE GLOBAL TEMPORARY <TAB

全面解析oracle中的锁机制3

5.TX事务锁和 TM(DML enqueue)锁: TX锁:数据块的前面有一个开销空间(ITL),这里会存放一个块的事务列表,对于每一个锁定了块的事务,都会在这个事务列表中有一个条目.这个结构的大小有创建对象时的两个参数决定: INITRANS:初始分配的可容纳事务的大小,一般初始为2,可以说是事务槽. MAXTRANS:这个结构可以扩缩到的最大大小.它默认为255,在实际中,最小值为2.在Oracle10g 中,这个设置已经废弃了,所以不再使用.这个版本中的MAXTRANS 总是255.