oracle如何合并多个sys_refcursor详解
⼀、背景
在数据开发中,有时你需要合并两个动态游标sys_refcursor。
开发⼀个存储过程PROC_A,这个过程业务逻辑相当复杂,代码篇幅较长。⼀段时间后要开发⼀个PROC_B,要⽤PROC_A同样的逻辑,⽽且在这个过程中,还要循环调⽤PROC_A这个过程。摆在你⾯前的有两个选择。
打开PL/SQL,仔细的读PROC_A这个过程,⼀直到明⽩了所有的逻辑,然后在⾃⼰的过程中重写这个逻辑。
直接复制PROC_A这个过的代码过来,多写极端。还是业界标准⼤法好
针对循环调⽤的,建⽴⼀个临时表,循环插⼊数据到临时表(但这⾥还有⼀个问题,每次返回的游标可能列都不相同,建⽴临时表就显得复杂了)
⼆、思路
经过搜索查询,到以下可⾏的⽅案
1. 序列化sys_refcursor为xml⽂档,ORACLE对xml⽀持还不错,12C已经有JSON格式了
2. 使⽤ORACLE xml解析的⽅法,对序列化的xml⽂档,添加、删除、修改
3. 转换为内存表,通过游标返回查询的结果
为此你需要掌握的知识有
使⽤ Dbms_Lob 个package操作clob类型数据,因为解析后的游标可能⽤varchar2是装不下的,帮助地址
acle/cd/E11882_01/timesten.112/e21645/d_lob.htm#TTPLP600。
重点掌握Oracle类型xmltype如何使⽤
acle/cd/B19306_01/appdev.102/b14258/t_xml.htm#BABHCHHJ
三、实现oracle游标的使用
从上边的帮助⽂档中,知道xmltype的构造函数中可以直接传⼊游标xmltype(refcursor)从⽽得到⼀个xmltype,调⽤xmltype的getClobVal⽅法,可得到序列化的结果,所以它的结构是这样的
<?xml version="1.0"?>
<ROWSET>
<ROW>
<COLUMNNAME1></COLUMNNAME1>
<COLUMNNAME2></COLUMNNAME2>
<...>...</...>
</ROW>
....
</ROWSET>
所以,如果需要合并两个数据列相同游标,只需要提取DOM中的ROW节点数据保存到定义的clob字段中去。
提取dom中⽚段,采⽤标准的xpath语法,/ROWSET/ROW这⾥提取ROW信息
Declare
x xmltype;
rowxml clob;
mergeXml clob;
ref_cur Sys_Refcursor;
ref_cur2 Sys_Refcursor;
ref_cur3 Sys_Refcursor;
begin
open ref_cur for
select F_USERNAME, F_USERCODE, F_USERID
from Tb_System_User
where F_userid = 1;
atetemporary(mergeXml, true);
Dbms_Lob.writeappend(mergeXml, 8, '<ROWSET>');
x := xmltype(ref_cur);
Dbms_Output.put_line('=====完整的REFCURSOR结构=====');
Dbms_Output.put_ClobVal());
Dbms_Output.put_line('=====只提取⾏信息=====');
rowxml := x.extract('/ROWSET/ROW').getClobVal(0, 0);
Dbms_Output.put_line(rowxml);
Dbms_Lob.append(mergeXml, rowxml);ROWSET
open ref_cur2 for
select F_USERNAME, F_USERCODE, F_USERID
from Tb_System_User
where F_userid = 1000;
x := xmltype(ref_cur2);
rowxml := x.extract('/ROWSET/ROW').getClobVal(0, 0);
Dbms_Lob.append(mergeXml, rowxml);
Dbms_Lob.writeappend(mergeXml, 9, '</ROWSET>');
Dbms_Output.put_line('=====合并后的信息=====');
Dbms_Output.put_line(mergeXml);
end;
执⾏这段代码输出的结果是这样的
=====完整的REFCURSOR结构=====
<?xml version="1.0"?>
<ROWSET>
<ROW>
<F_USERNAME>系统管理员</F_USERNAME>
<F_USERCODE>admin</F_USERCODE>
<F_USERID>1</F_USERID>
</ROW>
</ROWSET>
=====只提取⾏信息=====
<ROW>
<F_USERNAME>系统管理员</F_USERNAME>
<F_USERCODE>admin</F_USERCODE>
<F_USERID>1</F_USERID>
</ROW>
=====合并后的信息=====
<ROWSET><ROW>
<F_USERNAME>系统管理员</F_USERNAME>
<F_USERCODE>admin</F_USERCODE>
<F_USERID>1</F_USERID>
</ROW>
<ROW>
<F_USERNAME>黄燕</F_USERNAME>
<F_USERCODE>HUANGYAN</F_USERCODE>
<F_USERID>1000</F_USERID>
</ROW>
</ROWSET>
Dbms_Output.put_line(mergeXml);
open ref_cur3 for
select *
from xmltable('/ROWSET/ROW' Passing xmltype(mergeXml) Columns
F_USERNAME varchar2(100) path 'F_USERNAME',
F_USERCODE varchar2(100) path 'F_USERCODE');
简单说明下xmltable构造函数
声明xpath,指明你需要解析的dom在哪⾥,⽐如从根到ROW /ROWSET/ROW
指明你要查询的xmltype
定义转换列,⽐如把ROW下边的F_USERNAME这个节点值,映射到游标列F_USERNAME 这个列中
附:sys_refcursor 和 cursor 优缺点⽐较
优点⽐较
优点⼀:sys_refcursor,可以在存储过程中作为参数返回⼀个table格式的结构集(我把他认为是table类型,容易理解,其实是⼀个游标集), cursor 只能⽤在存储过程,函数,包等的实现体中,不能做参数使⽤。
优点⼆:sys_refcursor 这东西可以使⽤在包中做参数,进⾏数据库⾯向对象开放。哈哈。我喜欢。cursor就不能。
缺点⽐较:
缺点:sys_refcursor 不能⽤open,close ,fetch 进⾏操作。不好学,难理解。
cursor可以⽤ open,close ,fetch操作,容易学,易懂
四、总结
xml作为早期数据传输,序列化和反序列化的⽂件格式,在oracle中也有良好的⽀持。所以,对于基于语⾔之上的知识,各个
语⾔实现⽅式基本相识。基础终究是重要的。
好了,以上就是这篇⽂章的全部内容了,希望本⽂的内容对⼤家的学习或者⼯作具有⼀定的参考学习价值,如果有疑问⼤家可以留⾔交流,谢谢⼤家对的⽀持。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论