oracle如何合并多个sys_refcursor详解

 更新时间:2018年05月31日 08:37:31   作者:_herbert  
这篇文章主要给大家介绍了关于oracle如何合并多个sys_refcursor的相关资料,以及在文末跟大家分享了sys_refcursor 和 cursor 优缺点的比较,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面来一起学习学习吧

一、背景

在数据开发中,有时你需要合并两个动态游标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方法,可得到序列化的结果,所以它的结构是这样的

<?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操作,容易学,易懂

四、总结

xml作为早期数据传输,序列化和反序列化的文件格式,在oracle中也有良好的支持。所以,对于基于语言之上的知识,各个语言实现方式基本相识。基础终究是重要的。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

相关文章

  • Oracle存储过程和自定义函数详解

    Oracle存储过程和自定义函数详解

    本篇文章主要介绍了Oracle存储过程和自定义函数详解,有需要的可以了解一下。
    2016-11-11
  • Oracle中3种常用的分页查询方法

    Oracle中3种常用的分页查询方法

    这篇文章主要给大家介绍了关于Oracle中3种常用的分页查询方法,分页查询就是把query到的结果集按页显示,比如一个结果集有1W行,每页按100条数据库,而你获取了第2页的结果集,需要的朋友可以参考下
    2023-09-09
  • Oracle中的定时任务实例教程

    Oracle中的定时任务实例教程

    定时任务相信大家都不陌生,下面这篇文章主要给大家介绍了关于Oracle中定时任务的相关资料,文中通过实例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2023-04-04
  • oracle数据库tns配置方法详解

    oracle数据库tns配置方法详解

    TNS是Oracle Net的一部分,专门用来管理和配置Oracle数据库和客户端连接的一个工具,在大多数情况下客户端和数据库要通讯,必须配置TNS,下面看一如何配置它吧
    2013-12-12
  • Oracle 12c安装方法及一些使用问题

    Oracle 12c安装方法及一些使用问题

    这篇文章主要介绍了Oracle 12c安装方法及一些使用问题,需要的朋友可以参考下
    2016-09-09
  • oracle异常(预定义异常,自定义异常)应用介绍

    oracle异常(预定义异常,自定义异常)应用介绍

    在开发过程中,经常会遇到一些测试,这时候就会想了解测试的过程,然后再根据过程分析代码错在哪里,这种情况下,就需要用到自定义异常,需要了解的朋友可以参考本文
    2012-11-11
  • Oracle递归查询简单示例

    Oracle递归查询简单示例

    最近在做一个树状编码管理系统,其中用到了oracle的树状递归查询,下面这篇文章主要给大家介绍了关于Oracle递归查询的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-11-11
  • 浅谈入门级oracle数据库数据导入导出步骤

    浅谈入门级oracle数据库数据导入导出步骤

    这篇文章主要介绍了浅谈入门级oracle数据库数据导入导出步骤,文章通过步骤解析介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-08-08
  • 浅析Oracle中的不等于号

    浅析Oracle中的不等于号

    前几天碰到一个关于Oracle不等于的问题,最后搜索了一下,发现下面资料,拿来跟大家分享一下,需要的朋友可以参考下
    2013-07-07
  • Oracle学习笔记(六)

    Oracle学习笔记(六)

    最近需要用的oracle,所以大家好好的学习下基础并整理下资料,希望能帮助到需要的朋友。
    2011-12-12

最新评论