oracle如何合并多个sys_refcursor详解
一、背景
在数据开发中,有时你需要合并两个动态游标sys_refcursor。
开发一个存储过程proc_a,这个过程业务逻辑相当复杂,代码篇幅较长。一段时间后要开发一个proc_b,要用proc_a同样的逻辑,而且在这个过程中,还要循环调用proc_a这个过程。摆在你面前的有两个选择。
- 打开pl/sql,仔细的读proc_a这个过程,一直到明白了所有的逻辑,然后在自己的过程中重写这个逻辑 。
- 直接复制proc_a这个过的代码过来,多写极端。还是业界标准大法好
- 针对循环调用的,建立一个临时表,循环插入数据到临时表(但这里还有一个问题,每次返回的游标可能列都不相同,建立临时表就显得复杂了)
好吧,这个新的过程是完成了,可是看上去,它更复杂了,代码量更大了。完全不能接受,必须改改!
这时,已经默默打开了oracle官方帮助文档 https://docs.oracle.com/cd/b19306_01/index.htm,寻找一个可行的办法,最终目标标是要解析,整合,合并 游标 sys_refcursor
二、思路
经过搜索查询,找到以下可行的方案
- 序列化sys_refcursor为xml文档,oracle对xml支持还不错,12c已经有json格式了
- 使用oracle xml解析的方法,对序列化的xml文档,添加、删除、修改
- 转换为内存表,通过游标返回查询的结果
为此你需要掌握的知识有
- 使用 dbms_lob 个package操作clob类型数据,因为解析后的游标可能用varchar2是装不下的,帮助地址 https://docs.oracle.com/cd/e11882_01/timesten.112/e21645/d_lob.htm#ttplp600。
- 重点掌握oracle类型xmltype如何使用 https://docs.oracle.com/cd/b19306_01/appdev.102/b14258/t_xml.htm#babhchhj
三、实现
从上边的帮助文档中,知道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; dbms_lob.createtemporary(mergexml, true); dbms_lob.writeappend(mergexml, 8, '<rowset>'); x := xmltype(ref_cur); dbms_output.put_line('=====完整的refcursor结构====='); dbms_output.put_line(x.getclobval()); 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>
从上边打印的结果看,我们已经成功的将两个游标 ref_cur和ref_cur2中我们需要的列信息合并到了一个xml文档中。那么接下了,我们就需要通过解析这个xml并返回一个新的sys_refcursor,这里你有必要了解以下oracle xmltable的用法(https://docs.oracle.com/cd/b19306_01/server.102/b14200/functions228.htm)接上边代码
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中也有良好的支持。所以,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。