数据库Oracle存款和储蓄进度接受手艺与资历总括,使用游标进度中现身的荒诞

数据库

上边包车型大巴意见是在运用游标的长河中做的日记。作者也是首先次利用,若是有怎么样难堪的地点请争辨指正,大家一块使劲。

新近是因为职业亟待,轻易领悟了下SQL Server 二〇〇五数据库创立轻松的在积存进度。大器晚成、首先表达如何创造存款和储蓄进程: CREATE
PROCEDUER my_pro @inputDate varchar ,//申明输入变量 @Result varchar
output //证明输出变量 AS declare @variable1 varchar//表明varchar变量
declare @variable2 int //注脚整形变量 BEGIN IF … END ELSE BEGIN ….
END END 以上便是开创二个简约的蕴藏进程方法。

oracle存储进度常用本事

1.

二、删除存款和储蓄进度 DROP PROCEDURE my_pro

大家在开展pl/sql编制程序时打交道最多的便是积攒进程了。存款和储蓄进程的协会是十一分的简便的,大家在那除了读书存款和储蓄进度的主导构造外,还也许会学习编写存款和储蓄进程时有关的意气风发部分实用的学问。如:游标的拍卖,格外的拍卖,集结的选择等等

  消息 16951,级别 16,状态 1,过程
usp_proc,第 16 行
      变量 ‘@myref’
不能够用作参数,因为在举办该进度前,不得为 CU传祺SO奥迪Q7 OUTPUT
参数分配游标。

三、试行存款和储蓄进程 试行未有参数的蕴藏进度:EXECUTE my_pro
实施有参数的囤积进度:EXECUTE my_pro ‘输入变量’ 的蕴藏进程:declare
@Result varchar//注脚输出变量 EXECUTE my_pro ‘输入变量’ @Result output
//让该存款和储蓄进程输出值保存到@Result中 select @Result as
result//查询结果再次回到

1.积存进度构造

  这些标题是自己在调用二个递归的、输出cursor output
的积累进程

四、例子上边要求将查询多条记下结果,使用一条重返值实行再次来到。诸如:有表Test中,个中有多少个字段为Test01与Test02,把Test01
>
10000的笔录,让查询Test02结果进行合併,归并成一条记下来回到。则赶回结果为:WSXEDCEnclaveFVTGB.
Test01 Test02 10000 QAZ 10001 WSX 10002 EDC 10003 TiggoFV 10004 TGB
那么,使用存款和储蓄进程,使用存款和储蓄进程中的游标,能够实行for循环来拓宽多条记下的联结。将得以减轻这几个主题材料。切切实实存款和储蓄进度如下:
DROP PORCEDURE my_Cursor //习贯性,创造早先,先看看该存款和储蓄进程是还是不是留存
CREATE PROCEDURE my_Cursor @id int ,//注明输入变量@Result varchar
output //注解输出变量,注意必须求有output关键字,不然默感到输入变量 AS
Declare city_cursor cursor for –阐明游标变量 Select [Test01] from
Test where Test01 > @id –此中@id为输入变量 Set @Result = ‘’
–设置变量@Result值 Declare @Field int–表明有的时候存放查询到的Test01的变量
Open city_cursor–张开游标 Fetch next from city_cursor into @Field–将实际Test01赋给变量,进行循环 While–循环开头 BEGIN If @Result = ‘’
BEGIN Select @Result = Test02 from Test where Test01 = @Field END ELSE
BEGIN Select @Result = @Result + ‘,’ + Test02 from Test where Test01 =
@Field END Fetch next from city_cursor into @Field –循环下三个Test01
END Close city_cursor–关闭游标 Deallocate city_cursor –释放游标引用 GO
以下是推行 Declare @Result varchar EXECUTE my_pro 10000 ,@Result output
–output关键字一定得写,假若是多少个输入参数,则接纳“,”号来区分 Select
@Result as result –查询结果

1.1 第一个存款和储蓄进程

create proc usp_proc(
@level int
@myref cursor varying output
)
as
begin
    if @level=3
        begin
             set @myref=cursor local static for
            select * from table
            open @myref
        end
     if @level<3
        begin
        declare @cur cursor
        exec usp_proc 2 @cur output --递归
        --
        --对输出游标@cur做一些操作
        --
        --使用完游标
        close @cur  --关闭游标
        deallocate @cur --删除游标
        end
end            

create or replace procedure proc1(  

纵然未有对出口的游标做close、deallocate管理就能并发上边错误。

 p_para1 varchar2,  

2.

 p_para2 out varchar2,  

  未有为@cur,分配游标

 p_para3 in out varchar2  

  那一个主题材料是本人在使用存款和储蓄进程再次回到的游标 cursor
output 发生的

)as    

  

v_name varchar2(20);  

create proc myproc(
@mycur cursor varying output
)
as
begin
set @mycur=cursor local static  for
select * from table

open @mycur --打开游标
end

--调用myproc
declare @cur cursor
exec myproc @cur output
fetch next from @cur
while @@fetch_status=0
    begin
    --使用游标
    fetch next from @cur
    end 

begin  

并发上述错的缘由正是概念游标后要求开荒 open @mycur

 v_name := ‘张三丰’;  

 p_para3 := v_name;  

 dbms_output.put_line(‘p_para3:’||p_para3);  

end;  

地点正是三个最简易的囤积进程。三个囤积进程差不离分为这么多少个部分:

创制语句:create or replace procedure 存款和储蓄进程名

假定未有or
replace语句,则只是是新建二个仓储进程。倘若系统存在该存款和储蓄进度,则会报错。Create
or replace procedure
即使系统中并未此存储进度就新建叁个,借使系统中有此存款和储蓄进程则把原先删除掉,重新创建四个储存进程。

储存进度名定义:蕴涵存款和储蓄进度名和参数列表。参数名和参数类型。参数名不可能重复,
参数字传送递方式:IN, OUT, IN OUT

IN 表示输入参数,按值传递方式。

OUT
表示输出参数,能够驾驭为按援引传递情势。能够视作存款和储蓄进度的输出结果,供外界调用者使用。

IN OUT 就能够作输入参数,也可作输出参数。

参数的数据类型只必要指明类型名就能够,没有必要钦定宽度。

参数的大幅度由外界调用者决定。

经过能够有参数,也能够没有参数

变量评释块:紧跟着的as (is
卡塔尔关键字,能够领略为pl/sql的declare关键字,用于注解变量。

变量注脚块用于表明该存款和储蓄进程须要利用的变量,它的作用域为该存款和储蓄进程。此外这里表明的变量必需内定宽度。据守PL/SQL的变量注脚标准。

进度语句块:从begin
关键字开始为经过的语句块。存储进度的切实可行逻辑在此来达成。

不行管理块:关键字为exception ,为拍卖语句发生的丰裕。该有的为可选

利落块:由end关键字结果。

1.2 存款和储蓄进度的参数传递情势

存款和储蓄进程的参数传递有二种方法:IN,OUT,IN OUT .

IN
按值传递,並且它差异目的在于积攒进程中被重复赋值。即使存款和储蓄进度的参数未有一些名存参数字传送递类型,默以为IN

create or replace procedure proc1(  

 p_para1 varchar2,  

 p_para2 out varchar2,  

 p_para3 in out varchar2  

)as    

v_name varchar2(20);  

begin  

 p_para1 :=’aaa’;  

 p_para2 :=’bbb’;  

 v_name := ‘张三丰’;  

 p_para3 := v_name;  

 dbms_output.put_line(‘p_para3:’||p_para3);  

 null;  

end;  

     

Warning: Procedure created with compilation errors  

 

SQL> show error;  

Errors for PROCEDURE LIFEMAN.PROC1:  

 

LINE/COL ERROR  


———————————————————————-  

8/3      PLS-00363: expression ‘P_PARA1’ cannot be used as an
assignment target  

8/3      PL/SQL: Statement ignored  

那或多或少与其余高等语言都不可人己一视。它约等于java在参数后边加上final关键字。

OUT
参数:作为出口参数,需求注意,当三个参数被钦定为OUT类型时,尽管在调用存款和储蓄进度此前对该参数举行了赋值,在存款和储蓄进度中该参数的值仍为null.

create or replace procedure proc1(  

 p_para1 varchar2,  

 p_para2 out varchar2,  

 p_para3 in out varchar2  

)as    

v_name varchar2(20);  

begin  

 v_name := ‘张三丰’;  

 p_para3 := v_name;  

 dbms_output.put_line(‘p_para1:’||p_para1);  

 dbms_output.put_line(‘p_para2:’||p_para2);  

 dbms_output.put_line(‘p_para3:’||p_para3);  

end;  

 

SQL> var p1 varchar2(10);  

SQL> var p2 varchar2(10);  

SQL> var p3 varchar2(10);  

SQL> exec :p1 :=’aaaa’;  

SQL> exec :p2 :=’bbbb’;  

SQL> exec :p3 :=’cccc’;  

SQL> exec proc1(:p1,:p2,:p3);  

p_para1:aaaa  

p_para2:  

p_para3:张三丰  

SQL> exec dbms_output.put_line(:p2);  

 

 

PL/SQL procedure successfully completed  

p2  

———  

INOUT 是的确的按援引传递参数。就能够作为传播参数也得以看作传播参数。

1.3 存储进度参数宽度  

create or replace procedure proc1(  

 p_para1 varchar2,  

 p_para2 out varchar2,  

 p_para3 in out varchar2  

)as    

v_name varchar2(2);  

begin  

 v_name := p_para1;  

end;  

 

SQL> var p1 varchar2(10);  

SQL> var p2 varchar2(20);  

SQL> var p3 varchar2(30);  

SQL> exec :p1 :=’aaaaaa’;  

SQL> exec proc1(:p1,:p2,:p3);  

     

     

ORA-06502: PL/SQL: numeric or value error: character string buffer too
small  

ORA-06512: at “LIFEMAN.PROC1”, line 8  

ORA-06512: at line 1  

第豆蔻梢头,大家要知道,大家鞭比不上腹在积存进程的概念中钦定期存款款和储蓄参数的宽窄,也就变成了作者们高不可攀在存储进程中央调控制传入变量的肥瘦。那一个宽度是一心由外界传入时间调节制的。

我们再来看看OUT类型的参数的拉长率。

create or replace procedure proc1(  

 p_para1 varchar2,  

 p_para2 out varchar2,  

 p_para3 in out varchar2  

)as    

v_name varchar2(2);  

begin  

 p_para2 :=’aaaaaaaaaaaaaaaaaaaa’;  

end;  

SQL> var p1 varchar2(1);  

SQL> var p2 varchar2(1);  

SQL> var p3 varchar2(1);  

SQL> exec :p2 :=’a’;  

SQL> exec proc1(:p1,:p2,:p3);  

在该过程中,p_para2被授予了19个字符a.

而在外表的调用进程中,p2那么些参数仅仅被定义为varchar2(1卡塔尔(英语:State of Qatar).

而把p2作为参数调用那几个进度,却并未报错。何况它的真实值正是19个a

SQL> select dump(:p2) from dual;  

DUMP(:P2)  


 

Typ=1 Len=20:
97,97,97,97,97,97,97,97,97,97,97,97,97,97,97,97,97,97,97,97  

p2  

———  

aaaaaaaaaaaaaaaaaaaa  

     

   再来看看IN OUT参数的增幅  

create or replace procedure proc1(  

 p_para1 varchar2,  

 p_para2 out varchar2,  

 p_para3 in out varchar2  

)as    

v_name varchar2(2);  

begin  

 p_para3 :=’aaaaaaaaaaaaaaaaaaaa’;  

end;  

 

SQL> var p1 varchar2(1);  

SQL> var p2 varchar2(1);  

SQL> var p3 varchar2(1);  

SQL> exec proc1(:p1,:p2,:p3);  

施行那几个进程,仍旧正确实行。

足见,对于IN参数,其调幅是由外部调整。

对于OUT 和IN OUT 参数,其调幅是由存款和储蓄进程之中央调节制。

由此,在写存款和储蓄进程时,对参数的大幅度进行验证是特别有须要的,最明智的主意就是参数的数据类型使用%type。那样双方就高达了同生龙活虎。

1.3 参数的默许值

存款和储蓄进度的参数可以安装暗许值

create or replace procedure procdefault(p1 varchar2,  

                                       p2 varchar2 default ‘mark’)  

as    

begin  

 dbms_output.put_line(p2);  

end;  

 

SQL> set serveroutput on;  

SQL> exec procdefault(‘a’);  

mark

能够因此default
关键字为存款和储蓄进程的参数内定暗许值。在对存款和储蓄进度调用时,就足以省略暗许值。

亟需注意的是:默许值仅仅补助IN传输类型的参数。OUT 和 IN
OUT不可能内定暗中同意值

对于有暗许值的参数不是排在最后的气象。

create or replace procedure procdefault2(p1 varchar2 default ‘remark’,  

                                       p2 varchar2 )  

as    

begin  

 dbms_output.put_line(p1);  

end;  

首先个参数有暗中同意值,第三个参数未有。若是大家想利用第三个参数的默许值时

exec procdefault2(‘aa’);

如此那般是会报错的。

那怎么变呢?能够钦命参数的值。

SQL> exec procdefault2(p2 =>’aa’);  

remark

如此就OK了,钦赐aa传给参数p2

  1. 积累进程之中块

2.1 内部块

大家知晓了蕴藏进度的布局,语句块由begin初始,以end甘休。这么些块是可以嵌套。在语句块中得以嵌套任何以下的块。

Declare … begin … exception … end;  

create or replace procedure innerBlock(p1 varchar2)  

as    

 o1 varchar2(10) := ‘out1’;  

begin  

 dbms_output.put_line(o1);  

 declare    

   inner1 varchar2(20);  

 begin  

   inner1 :=’inner1′;  

   dbms_output.put_line(inner1);  

 

   declare    

     inner2 varchar2(20);  

   begin  

     inner2 := ‘inner2’;  

     dbms_output.put_line(inner2);  

   end;  

 exception    

   when others then  

     null;  

 end;  

end;  

须要小心变量的效能域。

3.积累进度的常用技艺

3.1 哪一种集结?

我们在利用存款和储蓄进程的时候时不时索要管理记录集,也正是多条数据记录。分为单列多行和多列多行,那几个项目都得以称作会集类型。我们在那间张开比较这几个集中类型,以便于在编程时做出精确的精选。

索引表,也叫做pl/sql表,不能积攒于数据库中,成分的个数未有约束,下标可感觉负值。

type t_table is table of varchar2(20) index by binary_integer;  

v_student t_table;  

varchar2(20卡塔尔表示存放成分的数据类型,binary_integer代表元素下标的数据类型。

嵌套表,索引表未有 index
by子句就是嵌套表,它能够存放于数据中,成分个数Infiniti,下标从1开端,而且须求初叶化

type t_nestTable is table of varchar2(20);  

v_class t_nestTable ;  

仅是这么评释是不能应用的,必需对嵌套表举行初阶化,对嵌套表进行初阶化能够接收它的结构函数

v_class :=t_nestTable(‘a’,’b’,’c’);  

变长数组,变长数组与高端语言的数组类型特别相似,下标以1发端,成分个数有限。

type t_array is varray (20) of varchar2(20);  

varray(20卡塔尔就定义了变长数组的最大因素个数是十多个

变长数组与嵌套表相通,也能够是数量表列的数据类型。

再者,变长数组的行使也须要事情发生早先开端化。

项目 可存款和储蓄于数据库 成分个数 是不是需开头化 发轫下标值

索引表 否 无限 不需

嵌套表 可 无限 需 1

可变数组 可 有限(自定义卡塔尔国 需 1

一句话来讲,假使单单是在积存进程中作为集结变量使用,索引表是最棒的挑肥拣瘦。

3.2 选取何种游标?

体现游标分为:普通游标,参数化游标和游标变量二种。

上边以几个历程来扩充认证

create or replace procedure proccursor(p varchar2)  

as    

v_rownum number(10) := 1;  

cursor c_postype is select pos_type from pos_type_tbl where rownum
=1;  

cursor c_postype1 is select pos_type from pos_type_tbl where rownum
= v_rownum;  

cursor c_postype2(p_rownum number) is select pos_type from
pos_type_tbl where rownum = p_rownum;  

type t_postype is ref cursor ;  

c_postype3 t_postype;  

v_postype varchar2(20);  

begin  

 open c_postype;  

 fetch c_postype into v_postype;  

 dbms_output.put_line(v_postype);  

 close c_postype;  

 open c_postype1;  

 fetch c_postype1 into v_postype;  

 dbms_output.put_line(v_postype);  

 close c_postype1;  

 open c_postype2(1);  

 fetch c_postype2 into v_postype;  

 dbms_output.put_line(v_postype);  

 close c_postype2;  

 open c_postype3 for select pos_type from pos_type_tbl where rownum
=1;  

 fetch c_postype3 into v_postype;  

 dbms_output.put_line(v_postype);  

 close c_postype3;  

end;  

cursor c_postype is select pos_type from pos_type_tbl where rownum
=1

这一句是概念了一个最平凡的游标,把全体查询已经写死,调用时不可能作任何改造。

cursor c_postype1 is select pos_type from pos_type_tbl where rownum
= v_rownum;

这一句并从未写死,查询参数由变量v_rownum来决定。供给小心的是v_rownum必得在此个游标定义早前宣称。

cursor c_postype2(p_rownum number) is select pos_type from
pos_type_tbl where rownum = p_rownum;

这一条语句与第二条效益相像,都以可觉得游标落成动态的询问。不过它越是的紧缩了参数的成效域范围。不过可读性减少了广大。

type t_postype is ref cursor ;

c_postype3 t_postype;

先定义了三个引用游标类型,然后再注解了贰个游标变量。

open c_postype3 for select pos_type from pos_type_tbl where rownum
=1;

然后再用open for
来开拓七个询问。需求介意的是它能够屡次应用,用来开拓差异的询问。

从动态性来讲,游标变量是最佳用的,但是阅读性也是最差的。

在乎,游标的定义只好用使重大字IS,它与AS不通用。

3.3 游标循环最好攻略

大家在进行PL/SQL编制程序时,日常须要循环读取结果集的数额。举办逐行管理,那个进度就需求对游标实行巡回。对游标举办巡回的措施有多种,大家在这里后生可畏一剖析。

create or replace procedure proccycle(p varchar2)  

as    

cursor c_postype is select pos_type, description from pos_type_tbl
where rownum < 6;  

v_postype varchar2(20);  

v_description varchar2(50);  

begin  

open c_postype;  

 if c_postype%found then  

   dbms_output.put_line(‘found true’);  

 elsif c_postype%found = false then  

   dbms_output.put_line(‘found false’);  

 else  

   dbms_output.put_line(‘found null’);  

 end if;  

 loop  

  fetch c_postype into v_postype,v_description ;  

  exit when c_postype%notfound;  

 
dbms_output.put_line(‘postype:’||v_postype||’,description:’||v_description);
 

 end loop;  

 close c_postype;  

dbms_output.put_line(‘—loop end—‘);  

 open c_postype;  

   fetch c_postype into v_postype,v_description;  

   while c_postype%found loop  

   
 dbms_output.put_line(‘postype:’||v_postype||’,description:’||v_description);
 

     fetch c_postype into v_postype,v_description ;  

   end loop;  

 

 close c_postype;  

dbms_output.put_line(‘—while end—‘);  

 for v_pos in c_postype loop  

   v_postype := v_pos.pos_type;  

   v_description := v_pos.description;  

 
 dbms_output.put_line(‘postype:’||v_postype||’,description:’||v_description);
 

 end loop;  

 dbms_output.put_line(‘—for end—‘);  

end;  

应用游标在此以前需求开打游标,open cursor,循环完后再关闭游标close cursor.

那是选择游标应该慎记于心的原理。

上边包车型地铁进度演示了游标循环的二种办法。

在讨论循环方法以前,我们先看看%found和%notfound那么些游标的性质。

open c_postype;  

if c_postype%found then  

  dbms_output.put_line(‘found true’);  

elsif c_postype%found = false then  

  dbms_output.put_line(‘found false’);  

else  

  dbms_output.put_line(‘found null’);  

end if;  

在开垦一个游标之后,立即检查它的%found或%notfound属性,它拿走的结果即不是true亦不是false.而是null.必得施行一条fetch语句后,这么些属性才有值。

先是种采用loop 循环

loop  

  fetch c_postype into v_postype,v_description ;  

  exit when c_postype%notfound;  

  ……  

end loop  

此地须要小心,exit when语句必定要紧跟在fetch之后。必制止多余的数量管理。

拍卖逻辑须要跟在exit when事后。这或多或少急需多加小心。

巡回甘休后要记得关闭游标。

第三种接收while循环。

  fetch c_postype into v_postype,v_description;  

while c_postype%found loop  

  ……  

     fetch c_postype into v_postype,v_description ;  

end loop;  

咱俩通晓了三个游标展开后,必需试行三次fetch语句,游标的属性才会起成效。所以接纳while
循环时,就需要在循环早先开展一遍fetch动作。

还要数量管理动作必得放在循环体内的fetch方法在此以前。循环体内的fetch方法要放在最终。不然就能够多管理三回。那或多或少也要那么些的小心。

说来讲去,使用while来循环处理游标是最复杂的情势。

第三种 for循环

for v_pos in c_postype loop  

  v_postype := v_pos.pos_type;  

  v_description := v_pos.description;  

  …  

end loop;  

可以知道for循环是比较容易实用的办法。

率先,它会自动open和close游标。消除了你忘记展开或关闭游标的沉闷。

其余,自动定义了多少个记下类型及证明该类型的变量,并自动fetch数据到这一个变量中。

咱俩需求专心v_pos
这几个变量不要求要在循环外实行宣示,没有需求要为其钦赐数据类型。

它应当是二个记下类型,具体的组织是由游标决定的。

其风姿浪漫变量的功用域仅仅是在循环体内。

把v_pos看作二个记录变量就足以了,要是要获得某二个值有如调用记录朝气蓬勃致就足以了。

如v_pos.pos_type

总的来讲,for循环是用来循环游标的最棒点子。高效,简洁,安全。

但缺憾的是,平日看见的却是第后生可畏种情势。所以自此得校正那些习贯了。

3.4 select into不可乎视的难题

大家知晓在pl/sql中要想从数据表中向变量赋值,须求运用select into 子句。

只是它会推动来部分主题材料,假诺查询未有记录时,会抛出no_data_found异常。

倘诺有多条记下时,会抛出too_many_rows异常。

其一是相比不佳的。生机勃勃旦抛出了非常,就能让进度中断。非常是no_data_found这种特别,未有严重到要让程序中断的地步,可以完全交给由程序进行拍卖。

create or replace procedure procexception(p varchar2)  

as    

 v_postype varchar2(20);  

begin  

  select pos_type into v_postype from pos_type_tbl where 1=0;  

   dbms_output.put_line(v_postype);  

end;  

     

实行这几个历程

SQL> exec procexception(‘a’);  

报错  

ORA-01403: no data found  

ORA-06512: at “LIFEMAN.PROCEXCEPTION”, line 6  

ORA-06512: at line 1  

管理这几个有多个主意

1. 平素抬Gott别处理。

create or replace procedure procexception(p varchar2)  

as    

 v_postype varchar2(20);  

   

begin  

  select pos_type into v_postype from pos_type_tbl where 1=0;  

   dbms_output.put_line(v_postype);  

exception    

 when no_data_found then  

   dbms_output.put_line(‘没找到数据’卡塔尔国;  

end;  

如此那般做换汤不换药,程序依然被中止。可能那样不是大家所想要的。

  1. select into做为一个独门的块,在此个块中张开足够管理

create or replace procedure procexception(p varchar2)  

as    

 v_postype varchar2(20);  

   

begin  

 begin  

  select pos_type into v_postype from pos_type_tbl where 1=0;  

   dbms_output.put_line(v_postype);  

exception    

 when no_data_found then  

   v_postype := ”;  

 end;  

 dbms_output.put_line(v_postype);  

end;  

那是风华正茂种对比好的管理方式了。不会因为这些可怜而孳生程序中断。

3.行使游标

create or replace procedure procexception(p varchar2)  

as    

 v_postype varchar2(20);  

 cursor c_postype is select pos_type  from pos_type_tbl where 1=0;  

begin  

 open c_postype;  

   fetch c_postype into v_postype;  

 close c_postype;  

 dbms_output.put_line(v_postype);  

end;  

那般就全盘的制止了no_data_found十分。完全交由工程师来开展调控了。

其次种景况是too_many_rows 相当的标题。

Too_many_rows 这几个主题材料比起no_data_found要复杂一些。

给叁个变量赋值时,不过查询结果有多个记录。

管理这种主题材料也会有两种状态:

1.
多条数据是能够选择的,也正是说从结果聚集随意取二个值就能够。这种状态应该很极端了吗,假如现身这种场地,也作证了前后相继的严俊性存在难点。

2.
多条数据是无法被选择的,在此种状态肯定是前后相继的逻辑出了难点,也说是说原本根本就不会想到它会生出多条记下。

对此第生机勃勃种情形,就不得不运用游标来拍卖,而对此第二种情景就亟须采用在这之中块来拍卖,重新抛出非常。

多条数据可以选用,随意取一条,那个跟no_data_found的管理方式相似,使用游标。

本人这里仅说第二种情状,不可承当多条数据,但是毫无忘了拍卖no_data_found哦。那就无法应用游标了,必需利用当中块。

create or replace procedure procexception2(p varchar2)  

as    

 v_postype varchar2(20);  

   

begin  

 begin  

   select pos_type into v_postype from pos_type_tbl where rownum
< 5;  

 exception  

   when no_data_found then  

     v_postype :=null;  

   when too_many_rows then  

   
 raise_application_error(-20000,’对v_postype赋值时,找到多条数据’卡塔尔(英语:State of Qatar);  

 end;  

dbms_output.put_line(v_postype);  

end;  

亟需留意的是不得不承认要丰裕对no_data_found的拍卖,对现身多条记下的气象则持续抛出特别,让上风姿罗曼蒂克层来拍卖。

一句话来讲对于select into的说话必要小心那二种情状了。须求妥善处理啊。

3.5 在存款和储蓄进度中回到结果集

大家利用存款和储蓄进程都以再次回到值都以单意气风发的,不经常大家须求从进程中回到贰个会师。即多条数据。这有两种减轻方案。比较轻松的做法是写不常表,不过这种做法不活络。况兼爱惜麻烦。我们得以应用嵌套表来完结.未有三个集中类型能够与java的jdbc类型相称。那正是指标与关周全据库的对抗吧。数据库的指标并不能统统调换为编制程序语言的靶子,还非得使用关周到据库的管理情势。

create or replace package procpkg is  

  type refcursor is ref cursor;  

  procedure procrefcursor(p varchar2, p_ref_postypeList  out
refcursor);  

end procpkg;  

 

create or replace package body procpkg is  

 procedure procrefcursor(p varchar2, p_ref_postypeList out  refcursor)
 

 is  

   v_posTypeList PosTypeTable;  

 begin  

   v_posTypeList :=PosTypeTable(卡塔尔(قطر‎;–开端化嵌套表  

   v_posTypeList.extend;  

   v_posTypeList(1卡塔尔(قطر‎ := PosType(‘A001’,’客商资料退换’卡塔尔(英语:State of Qatar);  

   v_posTypeList.extend;  

   v_posTypeList(2卡塔尔(英语:State of Qatar) := PosType(‘A002’,’团体资料退换’卡塔尔国;  

   v_posTypeList.extend;  

   v_posTypeList(3卡塔尔(英语:State of Qatar) := PosType(‘A003’,’收益人更换’卡塔尔(قطر‎;  

   v_posTypeList.extend;  

   v_posTypeList(4卡塔尔国 := PosType(‘A004’,’续期交费格局更换’卡塔尔国;  

   open p_ref_postypeList for  select * from table(cast
(v_posTypeList as PosTypeTable));  

 end;  

end procpkg;  

在德阳中定义了一个游标变量,并把它看做存款和储蓄进程的参数类型。

在仓库储存进度中定义了多个嵌套表变量,对数码写进嵌套表中,然后把嵌套表进行类型调换为table,游标变量从那些嵌套表中实行查询。外界程序调用那些游标。

据此那么些进程要求定义四个种类。

create or replace type PosType as Object (  

 posType varchar2(20),  

 description varchar2(50)  

);  

create or replace type PosTypeTable is table of PosType;

急需专心,那四个门类无法定义在常德中,必得独立定义,那样java层才干动用。

在表面通过pl/sql来调用这么些进度特轻巧。

set serveroutput on;  

declare    

 type refcursor is ref cursor;  

 v_ref_postype refcursor;  

 v_postype varchar2(20);  

 v_desc varchar2(50);  

begin  

 procpkg.procrefcursor(‘a’,v_ref_postype);  

 loop  

   fetch  v_ref_postype into v_postype,v_desc;  

   exit when v_ref_postype%notfound;  

   dbms_output.put_line(‘posType:’|| v_postype || ‘;description:’ ||
v_desc);  

 end loop;  

end;  

静心:对于游标变量,无法应用for循环来管理。因为for循环会隐式的施行open动作。而通过open
for来开拓的游标%isopen是为true的。也便是暗许打开的。Open叁个早已open的游标是不对的。所以不能够动用for循环来管理游标变量。

作者们第意气风发商讨的是何等通过jdbc调用来拍卖这一个输出参数。

conn = this.getDataSource().getConnection();  

CallableStatement call = conn.prepareCall(“{call
procpkg.procrefcursor(?,?)}”);  

call.setString(1, null);  

call.registerOutParameter(2, OracleTypes.CURSOR);  

call.execute();  

ResultSet rsResult = (ResultSet) call.getObject(2);  

while (rsResult.next()) {  

 String posType = rsResult.getString(“posType”);  

 String description = rsResult.getString(“description”);  

 ……  

}  

这正是jdbc的处理情势。

Ibatis管理办法:

1.参数配置

<parameterMap id=”PosTypeMAP” class=”java.util.Map”>    

<parameter property=”p” jdbcType=”VARCHAR”
javaType=”java.lang.String” />    

<parameter property=”p_ref_postypeList” jdbcType=”ORACLECURSOR”
javaType=”java.sql.ResultSet” mode=”OUT”
typeHandler=”com.palic.elis.pos.dayprocset.integration.dao.impl.CursorHandlerCallBack”
/>    

</parameterMap>  

 

2.调用经过  

 <procedure id =”procrefcursor” parameterMap =”PosTypeMAP”>  

     {call procpkg.procrefcursor(?,?)}  

 </procedure>  

 

3.概念自身的微处理器  

 public class CursorHandlerCallBack implements TypeHandler{  

   public Object getResult(CallableStatement cs, int index) throws
SQLException {  

       ResultSet rs = (ResultSet)cs.getObject(index);  

       List result = new ArrayList();  

       while(rs.next()) {  

           String postype =rs.getString(1);  

           String description = rs.getString(2);  

           CodeTableItemDTO posTypeItem = new CodeTableItemDTO();  

           posTypeItem.setCode(postype);  

           posTypeItem.setDescription(description);  

           result.add(posTypeItem);  

       }  

       return result;  

   }  

 

 

 

  1. dao方法  

   public List procPostype() {  

       String p = “”;  

       Map para = new HashMap();  

       para.put(“p”,p);  

       para.put(“p_ref_postypeList”,null);  

       
this.getSqlMapClientTemplate().queryForList(“pos_dayprocset.procrefcursor”,
 para);  

        return (List)para.get(“p_ref_postypeList”);  

   }  

其意气风发跟jdbc的方法充裕的相像.

大家利用的是ibatis的2.0本子,比较麻烦。

要是是应用2.2以上版本就特别轻巧的。

因为能够在parameterMap中定义一个resultMap.那样就无须要团结定义微处理器了。

能够从深入分析2.0和2.0的dtd文件知道。

地点的两种格局都以特别的复杂,假设单单是索要回到三个结实集,那就全盘能够利用函数来落到实处了。

create or replace package procpkg is  

  type refcursor is ref cursor;  

  procedure procrefcursor(p varchar2, p_ref_postypeList  out
refcursor);  

  function procpostype(p varchar2) return PosTypeTable;    

end procpkg;  

 

create or replace package body procpkg is  

 procedure procrefcursor(p varchar2, p_ref_postypeList out  refcursor)
 

 is  

   v_posTypeList PosTypeTable;  

 begin  

   v_posTypeList :=PosTypeTable(卡塔尔(英语:State of Qatar);–最初化嵌套表  

   v_posTypeList.extend;  

   v_posTypeList(1卡塔尔(قطر‎ := PosType(‘A001’,’客商资料改变’卡塔尔(قطر‎;  

   v_posTypeList.extend;  

   v_posTypeList(2) := PosType(‘A002’,’团体资料改变’卡塔尔;  

   v_posTypeList.extend;  

   v_posTypeList(3卡塔尔(قطر‎ := PosType(‘A003’,’受益人退换’卡塔尔国;  

   v_posTypeList.extend;  

   v_posTypeList(4卡塔尔国 := PosType(‘A004’,’续期交费形式更改’卡塔尔(英语:State of Qatar);  

   open p_ref_postypeList for  select * from table(cast
(v_posTypeList as PosTypeTable));  

 end;  

 

 function procpostype(p varchar2) return PosTypeTable  

 as  

  v_posTypeList PosTypeTable;  

 begin  

     v_posTypeList :=PosTypeTable(卡塔尔(قطر‎;–起初化嵌套表  

   v_posTypeList.extend;  

   v_posTypeList(1卡塔尔(قطر‎ := PosType(‘A001’,’客商资料更改’卡塔尔国;  

   v_posTypeList.extend;  

   v_posTypeList(2卡塔尔国 := PosType(‘A002’,’团体资料改动’卡塔尔(قطر‎;  

   v_posTypeList.extend;  

   v_posTypeList(3卡塔尔(قطر‎ := PosType(‘A003’,’受益人改造’卡塔尔(قطر‎;  

   v_posTypeList.extend;  

   v_posTypeList(4卡塔尔(英语:State of Qatar) := PosType(‘A004’,’续期交费形式退换’卡塔尔;  

   return  v_posTypeList;  

 end;  

end procpkg;  

ibatis配置

<resultMap id=”posTypeResultMap”
class=”com.palic.elis.pos.common.dto.CodeTableItemDTO”>  

  <result property=”code” column=”posType”/>  

  <result property=”description” column=”description”/>  

</resultMap>  

 

 <select id=”procPostype” resultMap=”posTypeResultMap”>  

   select * from table(cast (procpkg.procpostype(#value#) as
PosTypeTable))  

 </select>  

Dao的写法跟普通查询同后生可畏

public List queryPostype() {  

 return
this.getSqlMapClientTemplate().queryForList(“pos_dayprocset.procPostype”,
null);  

}  

有几点须要小心,这里不能够应用索引表,而是嵌套表。

其它就是把嵌套表强制转变为普通表。

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图