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文档,添加、删除、修改
- 转换为内存表,通过游标返回查询的结果
为此你需要掌握的知识有
- 使用 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
9 open ref_cur for
10 select F_USERNAME, F_USERCODE, F_USERID
from Tb_System_User
where F_userid = 1;
Dbms_Lob.createtemporary(mergeXml, true);
14 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 这个列中
四、总结
xml作为早期数据传输,序列化和反序列化的文件格式,在oracle中也有良好的支持。所以,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。
oracle 合并多个sys_refcursor的更多相关文章
- Sun 与 Oracle 合并的未来
引言 SUN 2008 初 10 亿美元收购 MySQL Oracle 2009 年 4 月 74 亿美元收购 SUN Sun 与 Oracle 合并的未来 1,如果云计算对企业来说变得越来越重要,那 ...
- oracle合并语句
在sql server中的合并语句可以用xml path 详见http://www.cnblogs.com/codeyu/archive/2010/05/25/1743474.html 而oracle ...
- Oracle合并函数内容
--MINUS去差集,取第一个集合有的而第二集合没有的,并以第一个字段排序select t.bumenbm from T_HQ_BM t minus select b.bumenbm from t_h ...
- Oracle ref cursor和sys_refcursor
1. 自定义 ref cursor 和 sys_refcursor; 2. sys_refcursor 做为参数传递结果集; 3. ref cursor 做为参数传递结果集; 1. 自定义 ref c ...
- Oracle合并某一列
本文转载自:https://www.cnblogs.com/LeiYang5237/p/6761183.html 一.oracle11g使用listagg() within group()函数 如图一 ...
- Oracle 合并查询
8).合并查询有时在实际应用中,为了合并多个select语句的结果,可以使用集合操作符号union,union all,intersect,minus.多用于数据量比较大的数据局库,运行速度快.1). ...
- oracle合并版本
1) 添加字段,并自增 第一步:alter table TOWN add ID int 第二步:Update TOWN set id=rownum; Commit; 2) 更新表(另一张表) ...
- oracle合并查询
1). Union 该操作符用于取得两个结果集的并集.当使用该操作符时,会自动去掉结果集中重复行. 2).union all 该操作符与union相似,但是它不会取消重复行,而且不会排序. 3). I ...
- Oracle 合并 merger into
merge into copy_emp1 c using employees e on (c.employee_id=e.employee_id)when matched then update ...
随机推荐
- django信号浅谈
Django中提供了“信号调度”,用于在框架执行操作时解耦.通俗来讲,就是一些动作发生的时候,信号允许特定的发送者去提醒一些接受者. 1.Django内置信号 Model signals pre_in ...
- Spring MVC基本概念
MVC工作原理: 模型-视图-控制器(MVC)是一个以设计界面应用程序为基础的设计模式.它主要通过分离模型.视图及控制器在应用程序中的角色将业务逻辑从界面中解耦.通常,模型负责封装应用程序数据在视图层 ...
- (转载)C# 枚举 FlagsAttribute用法
这是读过几篇文章后发现整理的最完整的一篇文章 转载地址:枚举特性FlagsAttribute的用法 先看官方的解释:指示可以将枚举作为位域(即一组标志)处理. 看起来并不好理解,到底什么是作为位域处理 ...
- hbase参数配置和说明
版本:0.94-cdh4.2.1 hbase-site.xml配置 hbase.tmp.dir 本地文件系统tmp目录,一般配置成local模式的设置一下,但是最好还是需要设置一下,因为很多文件都会默 ...
- Dalvik和ART
--摘自<Android进阶解密> DVM和ART都是在Zygote进程中诞生的 *DVM和JVM的区别* 1.基于的架构不同 DVM是基于寄存器的,它没有基于栈的虚拟机在复制数据时而使用 ...
- 执行JS
JS执行切换frame框架后点击 Window.frame['frame路径'].frame['下级frame路径'].document.getElementById("id"). ...
- 第六章 对象-javaScript权威指南第六版(二)
通过原型 继承创建一个新对象,对于这一个函数的有说不出的感觉,看看语句都很简单,深层次的东西就是不知道 function inherit(p) { if(p == null) throw TypeE ...
- 2018-2019-2 网络对抗技术 20162329 Exp2 后门原理与实践
目录 1.实践基础 1.1.什么是后门 1.2.基础问题 2.实践内容 2.1.使用netcat获取主机操作Shell,cron启动 2.2.使用socat获取主机操作Shell, 任务计划启动 2. ...
- ios 上下滑动粘滞问题
ios 移动端,当你触及到可以左右滑动部分,进行上下滑动操作时,会导致上下滑动粘滞卡顿的问题 mdn:https://developer.mozilla.org/zh-CN/docs/Web/CSS/ ...
- AT与ATX电源 - 1 系统状态
ATX与AT电源比较 ATX电源普遍应用在PC中,它有两套电源,一个是正常操作使用:12V,5V,3.3V和-12V,还有一个独立的5V待机电源,所谓的待机电源就是其ON的充要条件是AC输入存在,而正 ...