• 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

    二、思路

    经过搜索查询,找到以下可行的方案

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

    为此你需要掌握的知识有

    三、实现

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

    1 <?xml version="1.0"?>
    2 <ROWSET>
    3 <ROW>
    4 <COLUMNNAME1></COLUMNNAME1>
    5 <COLUMNNAME2></COLUMNNAME2>
    6 <...>...</...>
    7 </ROW>
    8 ....
    9 </ROWSET>

    所以,如果需要合并两个数据列相同游标,只需要提取DOM中的ROW节点数据保存到定义的clob字段中去。

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

     1 Declare
     2 x xmltype;
     3 rowxml clob;
     4 mergeXml clob;
     5 ref_cur Sys_Refcursor;
     6 ref_cur2 Sys_Refcursor;
     7 ref_cur3 Sys_Refcursor;
     8 begin
     9   open ref_cur for
    10    select F_USERNAME, F_USERCODE, F_USERID
    11    from Tb_System_User
    12    where F_userid = 1;
    13  Dbms_Lob.createtemporary(mergeXml, true);
    14  Dbms_Lob.writeappend(mergeXml, 8, '<ROWSET>');
    15  x := xmltype(ref_cur);
    16  Dbms_Output.put_line('=====完整的REFCURSOR结构=====');
    17  Dbms_Output.put_line(x.getClobVal());
    18  Dbms_Output.put_line('=====只提取行信息=====');
    19  rowxml := x.extract('/ROWSET/ROW').getClobVal(0, 0);
    20  Dbms_Output.put_line(rowxml);
    21  Dbms_Lob.append(mergeXml, rowxml);ROWSET
    22  open ref_cur2 for
    23   select F_USERNAME, F_USERCODE, F_USERID
    24   from Tb_System_User
    25   where F_userid = 1000;
    26  x := xmltype(ref_cur2);
    27  rowxml := x.extract('/ROWSET/ROW').getClobVal(0, 0);
    28  Dbms_Lob.append(mergeXml, rowxml);
    29  Dbms_Lob.writeappend(mergeXml, 9, '</ROWSET>');
    30  Dbms_Output.put_line('=====合并后的信息=====');
    31  Dbms_Output.put_line(mergeXml);
    32 end;

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

     1 =====完整的REFCURSOR结构=====
     2 <?xml version="1.0"?>
     3 <ROWSET>
     4 <ROW>
     5 <F_USERNAME>系统管理员</F_USERNAME>
     6 <F_USERCODE>admin</F_USERCODE>
     7 <F_USERID>1</F_USERID>
     8 </ROW>
     9 </ROWSET>
    10 
    11 =====只提取行信息=====
    12 <ROW>
    13 <F_USERNAME>系统管理员</F_USERNAME>
    14 <F_USERCODE>admin</F_USERCODE>
    15 <F_USERID>1</F_USERID>
    16 </ROW>
    17 
    18 =====合并后的信息=====
    19 <ROWSET><ROW>
    20 <F_USERNAME>系统管理员</F_USERNAME>
    21 <F_USERCODE>admin</F_USERCODE>
    22 <F_USERID>1</F_USERID>
    23 </ROW>
    24 <ROW>
    25 <F_USERNAME>黄燕</F_USERNAME>
    26 <F_USERCODE>HUANGYAN</F_USERCODE>
    27 <F_USERID>1000</F_USERID>
    28 </ROW>
    29 </ROWSET>

       从上边打印的结果看,我们已经成功的将两个游标 ref_curref_cur2中我们需要的列信息合并到了一个xml文档中。那么接下了,我们就需要通过解析这个xml并返回一个新的sys_refcursor,这里你有必要了解以下oracle xmltable的用法(https://docs.oracle.com/cd/B19306_01/server.102/b14200/functions228.htm)接上边代码

    1 Dbms_Output.put_line(mergeXml);
    2 open ref_cur3 for
    3   select *
    4   from xmltable('/ROWSET/ROW' Passing xmltype(mergeXml) Columns
    5         F_USERNAME varchar2(100) path 'F_USERNAME',
    6         F_USERCODE varchar2(100) path 'F_USERCODE'); 

    简单说明下xmltable构造函数

    • 声明xpath,指明你需要解析的dom在哪里,比如从根找到ROW /ROWSET/ROW
    • 指明你要查询的xmltype
    • 定义转换列,比如把ROW下边的F_USERNAME这个节点值,映射到游标列F_USERNAME 这个列中

    四、总结

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

  • 相关阅读:
    课后作业-阅读任务-阅读提问-3
    团队-团队编程项目作业名称-团队一阶段互评
    结对-四则运算答题器-结对项目总结
    课后作业-阅读任务-阅读笔记
    团队-科学计算器-开发文档
    团队编程项目作业6-程序维护
    课后作业-团队编程项目总结
    课后作业-阅读任务-阅读提问-4
    课后作业-阅读任务-阅读笔记-4
    个人编程作业2
  • 原文地址:https://www.cnblogs.com/yfrs/p/mergecursor.html
Copyright © 2020-2023  润新知