oracle like 合并,oracle 合并多个sys_refcursor

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文档,添加、删除、修改

转换为内存表,通过游标返回查询的结果

为此你需要掌握的知识有

三、实现

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

1 <?xml version="1.0"?>

2

3

4

5

6 <...>...

7

8 ....9

所以,如果需要合并两个数据列相同游标,只需要提取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 selectF_USERNAME, F_USERCODE, F_USERID11 fromTb_System_User12 where F_userid = 1;13 Dbms_Lob.createtemporary(mergeXml, true);14 Dbms_Lob.writeappend(mergeXml, 8, '');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);ROWSET22 open ref_cur2 for

23 selectF_USERNAME, F_USERCODE, F_USERID24 fromTb_System_User25 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, '');30 Dbms_Output.put_line('=====合并后的信息=====');31 Dbms_Output.put_line(mergeXml);32 end;

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

1 =====完整的REFCURSOR结构=====2 <?xml version="1.0"?>

3

4

5 系统管理员

6 admin

7 1

8

9

10

11 =====只提取行信息=====12

13 系统管理员

14 admin

15 1

16

17

18 =====合并后的信息=====19

20 系统管理员

21 admin

22 1

23

24

25 黄燕

26 HUANGYAN

27 1000

28

29

从上边打印的结果看,我们已经成功的将两个游标ref_cur和ref_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) Columns5 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中也有良好的支持。所以,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值