oracle 合并多个sys_refcursor

1、背景

在数据开发中,有时你须要合并两个动态游标sys_refcursorsql

开发一个存储过程PROC_A,这个过程业务逻辑至关复杂,代码篇幅较长。一段时间后要开发一个PROC_B,要用PROC_A一样的逻辑,并且在这个过程当中,还要循环调用PROC_A这个过程。摆在你面前的有两个选择。oracle

  • 打开PL/SQL,仔细的读PROC_A这个过程,一直到明白了全部的逻辑,而后在本身的过程当中重写这个逻辑 。
  • 直接复制PROC_A这个过的代码过来,多写极端。仍是业界标准大法好
  • 针对循环调用的,创建一个临时表,循环插入数据到临时表(但这里还有一个问题,每次返回的游标可能列都不相同,创建临时表就显得复杂了)

好吧,这个新的过程是完成了,但是看上去,它更复杂了,代码量更大了。彻底不能接受,必须改改!
这时,已经默默打开了ORACLE官方帮助文档 https://docs.oracle.com/cd/B1...,寻找一个可行的办法,最终目标标是要解析,整合,合并 游标 sys_refcursorapp

2、思路

通过搜索查询,找到如下可行的方案dom

  1. 序列化sys_refcursor为xml文档,ORACLE对xml支持还不错,12C已经有JSON格式了
  2. 使用ORACLE xml解析的方法,对序列化的xml文档,添加、删除、修改
  3. 转换为内存表,经过游标返回查询的结果

为此你须要掌握的知识有函数

3、实现

从上边的帮助文档中,知道xmltype的构造函数中能够直接传入游标xmltype(refcursor)从而获得一个xmltype,调用xmltype的getClobVal方法,可获得序列化的结果,因此它的结构是这样的code

<?xml version="1.0"?>
<ROWSET>
  <ROW>
    <COLUMNNAME1></COLUMNNAME1>
    <COLUMNNAME2></COLUMNNAME2>
    <...>...</...>
  </ROW>
  ....
</ROWSET>

因此,若是须要合并两个数据列相同游标,只须要提取DOM中的ROW节点数据保存到定义的clob字段中去。server

提取dom中片断,采用标准的xpath语法,/*/ROW 这里提取ROW信息xml

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;

执行这段代码输出的结果是这样的htm

=====完整的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/B1...内存

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 这个列中

4、总结

xml做为早期数据传输,序列化和反序列化的文件格式,在oracle中也有良好的支持。因此,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。

相关文章
相关标签/搜索