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

二、思路

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

  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中也有良好的支持。所以,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。

原文地址:https://www.cnblogs.com/yfrs/p/mergecursor.html

时间: 2024-11-06 19:37:01

oracle 合并多个sys_refcursor的相关文章

Oracle合并函数内容

--MINUS去差集,取第一个集合有的而第二集合没有的,并以第一个字段排序select t.bumenbm from T_HQ_BM t minus select b.bumenbm from t_hq_bm2 b --INTERSECT 取交集,取两个集合都有的,并按第一个字段排序 select t.bumenbm from T_HQ_BM t intersect select b.bumenbm from t_hq_bm2 b --UNION去重合并,去重复记录,并按第一个字段排序 sele

Oracle ref cursor和sys_refcursor

1. 自定义 ref cursor 和 sys_refcursor; 2. sys_refcursor 做为参数传递结果集; 3. ref cursor 做为参数传递结果集;   1. 自定义 ref cursor 和 sys_refcursor: declare type df_ref is ref cursor; --定义 ref cursor rf df_ref; --声明 rf 是df_ref ename varchar2(30); begin open rf for 'select e

Oracle 合并多行记录为一行

1.实际需求情况如下:    描述:上述表中,某一位同学有多门课程和成绩,现在希望有一条SQL语句,将课程名称和成绩合并在一起显示,如下: 描述:将这位同学的全部课程和成绩用一个字段表示,里面包括全部的课程名称和考试成绩 2.问题分析 解决这个问题可以采用创建function的办法,去合并这2个字段,但这样太麻烦,由于是采用的Oracle10g,故可以采用函数wmsys.wm_concat()实现此功能. 3.解决方案 SQL: 第一步先写一个视图view_ly_sjdxpt_jwcj,合并这2

oracle合并版本

1)   添加字段,并自增 第一步:alter table TOWN add ID int 第二步:Update TOWN set id=rownum; Commit; 2)   更新表(另一张表) Update a set(a.province,a.city)= (select province,city from b where b.mobile=a.mobile) 3)   命令行导出表 exp userid=账户/密码 tables=(TOWN,YIZHI,一带一路企业) file=C:

关于Oracle 合并字段

<select id="I0203F0004SQL001" parameterType="com.dsunsoft.cqhzz.sql.dto.I0203.I0203F0004SQL001DbcIU001" resultType="com.dsunsoft.cqhzz.sql.dto.I0203.I0203F0004SQL001DbcOU001"> SELECT t1.USER_ID AS userId, t1.USER_NM AS

Oracle 表复杂查询之多表合并查询

转自:https://www.cnblogs.com/GreenLeaves/p/6635887.html 本文使用到的是oracle数据库scott方案所带的表,scott是oracle数据库自带的方案,使用前请确保其解锁 Oracle合并查询一共有四种方式,分别使用不同的关键字:UNION.UNION ALL.MINUS.INTERSECT 1.UNION ALL 使用UNION ALL,表示取A.B的合集,不过滤重复的数据行,代码如下: select * from emp where sa

oracle创建存储过程并返回结果集(附C#调用代码)

使用存储过程中,最常用的莫过于查询数据表,并返回结果集. 在SQL SERVER 中,这类操作最简单,通过简单的select * from xx 即可完成.但是在Oracle中并不支持这种写法,那么我们怎么实现跟SQL SERVER同样的功能呢?且看以下代码: create or replace procedure sp_getdept (rep_type in varchar2,sel in varchar2,result out sys_refcursor) as seq varchar2(

Oracle hash分区的秘密

转自:http://www.hellodb.net/2009/12/hash_partition.html 在面试时经常会问一个问题,请列举出hash在数据库内部的应用,hash的原理虽然简单,但是它在数据库中可以说是无处不在.其中hash partition是hash在数据库中一个简单的应用,虽然它没有range partition那么常用,但是我们在做数据库水平拆分时,其实就是利用了hash partition的原理,利用hash函数对某个key进行运算,然后将其分布到不同的主机上,原理很简

Oracle查询转换之视图合并

一.简单视图合并:指针对那些不含外连接,以及所带视图定义sql语句中不含distinct,group by等聚合函数的目标sql的视图合并. create or replace view view_1 as SELECT t2.prod_id   FROM sales t2, customers t3  WHERE t2.cust_id = t3.cust_id    AND t3.cust_gender = 'M'; 视图合并:   SELECT t1.prod_id, t1.prod_nam