一、背景
在数据开发中,有时你需要合并两个动态游标sys_refcursor。
开发一个存储过程PROC_A,这个过程业务逻辑相当复杂,代码篇幅较长。一段时间后要开发一个PROC_B,要用PROC_A同样的逻辑,而且在这个过程中,还要循环调用PROC_A这个过程。摆在你面前的有两个选择。
好吧,这个新的过程是完成了,可是看上去,它更复杂了,代码量更大了。完全不能接受,必须改改!
这时,已经默默打开了ORACLE官方帮助文档 https://docs.oracle.com/cd/B19306_01/index.htm,寻找一个可行的办法,最终目标标是要解析,整合,合并 游标 sys_refcursor
二、思路
经过搜索查询,找到以下可行的方案
为此你需要掌握的知识有
三、实现
从上边的帮助文档中,知道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构造函数
附:sys_refcursor 和 cursor 优缺点比较
优点比较
优点一:sys_refcursor,可以在存储过程中作为参数返回一个table格式的结构集(我把他认为是table类型,容易理解,其实是一个游标集), cursor 只能用在存储过程,函数,包等的实现体中,不能做参数使用。
优点二:sys_refcursor 这东西可以使用在包中做参数,进行数据库面向对象开放。哈哈。我喜欢。cursor就不能。
缺点比较:
缺点:sys_refcursor 不能用open,close ,fetch 进行操作。不好学,难理解。
cursor可以用 open,close ,fetch操作,容易学,易懂
四、总结
xml作为早期数据传输,序列化和反序列化的文件格式,在oracle中也有良好的支持。所以,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对小牛知识库的支持。
问题内容: 如何将其写成一行? 用id和class选择html标签的方法是什么? 问题答案: 您所要做的就是用逗号分隔它们,例如
问题内容: 我的包体内有一个proc: 我在包装规格中公开了这一点。 如何在PL SQL Developer(或类似版本)的新SQL窗口中执行此过程? 问题答案: 您可以相对轻松地执行该过程 当然,这只是将游标返回到调用应用程序。它不执行任何操作来从游标中获取数据,对该数据执行任何操作或关闭游标。假设您的目标是向其中写入一些数据(有时对于原型设计很有用,但生产代码不应该依赖该数据),则可以执行以下
本文向大家介绍如何合并多个 .NET 程序集,包括了如何合并多个 .NET 程序集的使用技巧和注意事项,需要的朋友参考一下 任何曾今寻找过一个将多个程序合并到一个文件中的解决方案的人,可能都听说过项 ILMerge, SmartAssembly, 等这样的工具. 另外一个知名的方案是将DLL作为资源进行嵌入(如果感兴趣,这里有一篇描述这一方案的精彩文章:从嵌入的资源出载入DLL[^]). 而在某些
问题内容: 我有3个文本字段,其中用户键入表名和2个需要合并的列名。 如何将2列值合并(合并)为1? 我使用oracle 11g企业版 问题答案: 串联?
问题内容: 我正在使用’Lucene.Net’库,并且有以下问题。 不是创建两个单独的QueryParser对象并使用它们来获得两个Hits对象,而是可以使用一个QueryParser对象在两个字段上进行搜索,这样我只有一个Hits对象,它可以为我提供每个文档的总体得分? 问题答案: 有3种方法可以做到这一点。 第一种方法是手动构造查询,这是QueryParser内部的工作。这是执行此操作最强大的
问题内容: 供参考的是代码。我正在尝试制作一个记录到elasticsearch的hubot插件,然后使用hubot命令搜索那些日志。 https://gist.github.com/4050748 我正在尝试检索与两个查询匹配的记录。 我期待: 多达50条记录 具有给定用户的记录 最近一小时的记录 我有: 最多10条记录 具有给定用户的记录 从任何时候 我如何在过去一小时内获取带有某些用户名的所有
本文向大家介绍如何用python合并多个excel文件,包括了如何用python合并多个excel文件的使用技巧和注意事项,需要的朋友参考一下 安装模块 1、找到对应的模块 http://www.python-excel.org/ 2、用 pip install 安装 因为使用的是 python3,所以安装的时候采用 pip3。 XlsxWriter 示例 先看看简单的演示: 运行结果如下: 合
问题内容: 我有一个这样的表来保存体检的结果以及发送报告的日期和结果。实际上,发送的日期是基于临床访问日期。客户可以拥有一个或多个报告(日期可能有所不同) 我想从上述数据中提取以下报告。 我正在研究Postgresql。“交叉表”功能在这里无法使用,因为每个客户端的“ date_sent”不一致。 任何人都可以粗略地提出一个应如何查询的想法吗? 问题答案: 我建议采用以下方法: 它的输出格式不完全