上面包车型地铁观点是在利用游标的经过中做的日记。小编也是率先次选拔,借使有啥样难堪的地点请争辩指正,大家一块儿努力。

最近出于职业急需,简单询问了下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’
不能够用作参数,因为在实施该进程前,不得为 CUTiggoSOTucson 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结果举办统风流倜傥,归总成一条记下来回到。则赶回结果为:WSXEDC冠道FVTGB.
Test01 Test02 10000 QAZ 10001 WSX 10002 EDC 10003 EvoqueFV 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 =
@菲尔德 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被予以了21个字符a.

而在外界的调用进度中,p2这么些参数仅仅被定义为varchar2(1卡塔尔.

而把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卡塔尔国就定义了变长数组的最大体素个数是19个

变长数组与嵌套表相仿,也足以是数码表列的数据类型。

况兼,变长数组的选用也急需事前起头化。

类型 可存款和储蓄于数据库 成分个数 是还是不是需开头化 初阶下标值

索引表 否 无限 不需

嵌套表 可 无限 需 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;  

     

推行那些进度

home88一必发,SQL> exec procexception(‘a’);  

报错  

ORA-01403: no data found  

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

ORA-06512: at line 1  

管理那么些有多少个方式

1. 直接助长特别处理。

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’,’顾客资料改造’卡塔尔(قطر‎;  

   v_posTypeList.extend;  

   v_posTypeList(2卡塔尔(قطر‎ := PosType(‘A002’,’团体资料更改’卡塔尔(英语:State of Qatar);  

   v_posTypeList.extend;  

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

   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卡塔尔(英语:State of Qatar) := PosType(‘A001’,’顾客资料改造’卡塔尔(قطر‎;  

   v_posTypeList.extend;  

   v_posTypeList(2卡塔尔国 := 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;  

 

 function procpostype(p varchar2) return PosTypeTable  

 as  

  v_posTypeList PosTypeTable;  

 begin  

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

   v_posTypeList.extend;  

   v_posTypeList(1卡塔尔(英语:State of Qatar) := PosType(‘A001’,’顾客资料改变’卡塔尔(英语:State of Qatar);  

   v_posTypeList.extend;  

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

   v_posTypeList.extend;  

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

   v_posTypeList.extend;  

   v_posTypeList(4卡塔尔(قطر‎ := 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);  

}  

有几点要求专心,这里无法选择索引表,而是嵌套表。

其它正是把嵌套表强逼转变为普通表。

相关文章