oracle合并多个sys

56次阅读
没有评论

共计 3456 个字符,预计需要花费 9 分钟才能阅读完成。

自动写代码机器人,免费开通

这篇文章给大家分享的是有关 oracle 合并多个 sys_refcursor 的案例的内容。丸趣 TV 小编觉得挺实用的,因此分享给大家做个参考,一起跟随丸趣 TV 小编过来看看吧。

一、背景

在数据开发中,有时你需要合并两个动态游标 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 操作,容易学,易懂

感谢各位的阅读!关于“oracle 合并多个 sys_refcursor 的案例”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!

向 AI 问一下细节

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2023-12-04发表,共计3456字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)