关于dbms_sql的使用

PL/SQL中使用动态SQL编程

在PL/SQL程序设计过程中,会遇到很多必须使用动态sql的地方,oracle系统所提供的DMBS_SQL包可以帮助你解决问题。
(一)介绍
DBMS_SQL系统包提供了很多函数及过程,现在简要阐述其中使用频率较高的几种:

function open_cursor:打开一个动态游标,并返回一个整型;

procedure close_cursor(c in out integer);关闭一个动态游标,参数为open_cursor所打开的游标;

procedure parse(c in integer, statement in varchar2, language_flag in integer):对动态游标所提供的sql语句进行解析,参数C表示游标,statement为sql语句,language-flag为解析sql语句所用oracle版本,一般有V6,V7跟native(在不明白所连database版本时,使用native);

procedure define_column(c in integer, position in integer, column any datatype, [column_size in integer]):定义动态游标所能得到的对应值,其中c为动态游标,positon为对应动态sql中的位置(从1开始),column为该值所对应的变量,可以为任何类型,column_size只有在column为定义长度的类型中使用如VARCHAR2,CHAR等(该过程有很多种情况,此处只对一般使用到的类型进行表述);

function execute(c in integer):执行游标,并返回处理一个整型,代表处理结果(对insert,delete,update才有意义,而对select语句而言可以忽略);

function fetch_rows(c in integer):对游标进行循环取数据,并返回一个整数,为0时表示已经取到游标末端;

procedure column_value(c in integer, position in integer, value):将所取得的游标数据赋值到相应的变量,c为游标,position为位置,value则为对应的变量;

procedure bind_variable(c in integer, name in varchar2, value):定义动态sql语句(DML)中所对应字段的值,c为游标,name为字段名称,value为字段的值;

以上是在程序中经常使用到的几个函数及过程,其他函数及过程请参照oracle所提供定义语句dbmssql.sql

(二)一般过程
对于一般的select操作,如果使用动态的sql语句则需要进行以下几个步骤:
open cursor---&gtparse---&gtdefine column---&gtexcute---&gtfetch rows---&gtclose cursor;
而对于dml操作(insert,update)则需要进行以下几个步骤:
open cursor---&gtparse---&gtbind variable---&gtexecute---&gtclose cursor;
对于delete操作只需要进行以下几个步骤:
open cursor---&gtparse---&gtexecute---&gtclose cursor;

(三)具体案例

--第一步:2个测试用的表
SQL> create table sun_part
2 (area number(3),
3 name varchar2(32)
4 )
5 partition by range (area)
6 (
7 partition part_530 values less than(531),
8 partition part_531 values less than(532),
9 partition part_532 values less than(533),
10 partition part_max values less than(999)
11 );

表已创建。

SQL> create table sun_part_his
2 (region number(3),
3 name varchar2(32),
4 deal_date date default sysdate
5 );

表已创建。

--第二步:插入几条测试数据
SQL> insert into sun_part values(530,"HZ");
已创建 1 行。

SQL> insert into sun_part values(531,"JN");已创建 1 行。

SQL> insert into sun_part values(532,"QD");已创建 1 行。

SQL> insert into sun_part values(571,"HangZhou");已创建 1 行。

SQL> commit;提交完成。

SQL> select * from sun_part;
AREA NAME
---------- --------------------------------
530 HZ
531 JN
532 QD
571 HangZhou

SQL>

第三步:定义sun_demo过程

SQL> create or replace procedure sun_demo(p_area in varchar2) IS
2 v_sql varchar2(2000);
3
4 v_CursorID number;
5 v_CursorRET number;
6
7 v_rowid varchar2(20);
8 v_area number(3);
9 v_name varchar2(32);
10
11 begin
12 --
判断p_area参数是否有效

13 if lower(p_area) not in ("530","531","532","max") then
14 raise_application_error(-20001,"Warning:Argument Invalid");
15 end if;
16
17 v_sql:="select rowid,area,name from sun_part partition(part_"||p_area||")";

18 v_CursorID:=DBMS_SQL.OPEN_CURSOR;
19
20 DBMS_SQL.PARSE(v_CursorID, v_sql, DBMS_SQL.NATIVE);

21
22 dbms_sql.define_column(v_CursorID, 1, v_rowid,20);
23 dbms_sql.define_column(v_CursorID, 2, v_area);
24 dbms_sql.define_column(v_CursorID, 3, v_name,32);
25
26 v_CursorRET:=DBMS_SQL.EXECUTE(v_CursorID);
27 loop
28 if dbms_sql.fetch_rows(v_CurSorID) = 0 then
29 dbms_sql.close_cursor(v_CurSorID);
30 exit;
31 end if;
32
33 dbms_sql.column_value(v_CurSorID,1,v_rowid);
34 dbms_sql.column_value(v_CurSorID,2,v_area);
35 dbms_sql.column_value(v_CurSorID,3,v_name);
36 insert into sun_part_his values(v_area,v_name,sysdate);
37 delete from sun_part where rowid=v_rowid;
38 end loop;
39 commit;
40 end sun_demo;
41 /

过程已创建。

SQL>

第四步:测试验证

SQL> desc sun_demo;
PROCEDURE sun_demo
参数名称 类型 输入/输出默认值?
------------------------------ ----------------------- ------ --------
P_AREA VARCHAR2 IN

SQL> select * from sun_part_his;

未选定行

SQL> select * from sun_part;

AREA NAME
---------- --------------------------------
530 HZ
531 JN
532 QD
571 HangZhou

SQL> exec sun_demo("530");

PL/SQL 过程已成功完成。

SQL> select * from sun_part_his;

REGION NAME DEAL_DATE
---------- -------------------------------- ----------
530 HZ 29-5
-06

SQL> select * from sun_part;

AREA NAME
---------- --------------------------------
531 JN
532 QD
571 HangZhou

SQL> exec sun_demo("532");

PL/SQL 过程已成功完成。

SQL> select * from sun_part_his;

REGION NAME DEAL_DATE
---------- -------------------------------- ----------
530 HZ 29-5
-06
532 QD 29-5
-06

SQL> select * from sun_part;

AREA NAME
---------- --------------------------------
531 JN
571 HangZhou

SQL>
--
异常测试[建立sun_demo的异常判断,仅仅是为了演示
]
SQL> exec sun_demo("021");
BEGIN sun_demo("021"); END;

*
ERROR
位于第 1
:
ORA-20001: Warning:Argument Invalid
ORA-06512:
"STUDY.SUN_DEMO", line 14
ORA-06512:
line 1


(四)注意问题:
1,在整个程序的设计过程中,对游标的操作切不可有省略的部分,一旦省略其中某一步骤,则会程序编译过程既告失败,如在程序结尾处未对改游标进行关闭操作,则在再次调用过程时会出现错误.
2,dbms_sql除了可以做一般的select,insert,update,delete等静态的sql做能在过程中所做工作外,还能执行create等DDL操作,不过在执行该类操作时应首先显式赋予执行用户相应的系统权限,比如create table等.该类操作只需open cursor---&gtprase---&gtclose cursor即能完成.


来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/77580/viewspace-212778/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/77580/viewspace-212778/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值