百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 优雅编程 > 正文

三个经典实例助你掌握Oracle数据库bulk collect批量绑定

sinye56 2024-10-04 23:42 2 浏览 0 评论

概述

BULK COLLECT 子句会批量检索结果,即一次性将结果集绑定到一个集合变量中,并从SQL引擎发送到PL/SQL引擎。通常可以在SELECT INTO、 FETCH INTO以及RETURNING INTO子句中使用BULK COLLECT。


语法

FETCH BULK COLLECT <cursor_name> BULK COLLECT INTO <collection_name>
LIMIT <numeric_expression>;
or
FETCH BULK COLLECT <cursor_name> BULK COLLECT INTO <array_name>
LIMIT <numeric_expression>;

Oracle8i中首次引入了Bulk Collect特性,该特性可以让我们在PL/SQL中能使用批查询,批查询在某些情况下能显著提高查询效率。

采用bulk collect可以将查询结果一次性地加载到collections中。

而不是通过cursor一条一条地处理。

可以在select into,fetch into,returning into语句使用bulk collect。

注意在使用bulk collect时,所有的into变量都必须是collections


实例:BULK COLLECT将得到的结果集绑定到记录变量中

DECLARE
 TYPE emp_rec_type IS RECORD --声明记录类型
 ( 
 empno emp.empno%TYPE 
 ,ename emp.ename%TYPE
 ,hiredate emp.hiredate%TYPE 
); 
 TYPE nested_emp_type IS TABLE OF emp_rec_type; --声明记录类型变量
 emp_tab nested_emp_type;
 
BEGIN
 SELECT empno, ename, hiredate BULK COLLECT INTO emp_tab --使用BULK COLLECT 将所得的结果集一次性绑定到记录变量emp_tab中
 FROM emp;
 
 FOR i IN emp_tab.FIRST .. emp_tab.LAST 
 LOOP
 DBMS_OUTPUT.put_line('Current record is '||emp_tab(i).empno||chr(9)||emp_tab(i).ename||chr(9)||emp_tab(i).hiredate); 
 END LOOP;
END; 
/

实验:使用LIMIT限制FETCH数据量

在使用BULK COLLECT 子句时,对于集合类型,如嵌套表,联合数组等会自动对其进行初始化以及扩展(如下示例)。因此如果使用BULK

COLLECT子句操作集合,则无需对集合进行初始化以及扩展。由于BULK COLLECT的批量特性,如果数据量较大,而集合在此时又自动扩展,为避免过大的数据集造成性能下降,因此使用limit子句来限制一次提取的数据量。limit子句只允许出现在fetch操作语句的批量中。

DECLARE 
 CURSOR emp_cur IS SELECT empno, ename, hiredate FROM emp;
 TYPE emp_rec_type IS RECORD 
 ( 
 empno emp.empno%TYPE 
 ,ename emp.ename%TYPE
 ,hiredate emp.hiredate%TYPE
 ); 
 TYPE nested_emp_type IS TABLE OF emp_rec_type; -->定义了基于记录的嵌套表 
 
 emp_tab nested_emp_type; -->定义集合变量,此时未初始化 
 v_limit PLS_INTEGER := 5; -->定义了一个变量来作为limit的值 
 v_counter PLS_INTEGER := 0; 
 
BEGIN 
 OPEN emp_cur; 
LOOP 
 FETCH emp_cur BULK COLLECT INTO emp_tab -->fetch时使用了BULK COLLECT子句 
 LIMIT v_limit; -->使用limit子句限制提取数据量 
 EXIT WHEN emp_tab.COUNT = 0; -->注意此时游标退出使用了emp_tab.COUNT,而不是emp_cur%notfound 
 v_counter := v_counter + 1; -->记录使用LIMIT之后fetch的次数 
 FOR i IN emp_tab.FIRST .. emp_tab.LAST 
 LOOP 
 DBMS_OUTPUT.put_line( 'Current record is '||emp_tab(i).empno||CHR(9)||emp_tab(i).ename||CHR(9)||emp_tab(i).hiredate); 
 END LOOP; 
END LOOP; 
CLOSE emp_cur; 
DBMS_OUTPUT.put_line( 'The v_counter is ' || v_counter ); 
END;
/

实验:RETURNING 子句的批量绑定

BULK COLLECT除了与SELECT,FETCH进行批量绑定之外,还可以与INSERT,DELETE,UPDATE语句结合使用。当与这几个DML语句结合时,我们 需要使用RETURNING子句来实现批量绑定。

DECLARE 
 TYPE emp_rec_type IS RECORD 
 ( 
 empno emp.empno%TYPE 
 ,ename emp.ename%TYPE 
 ,hiredate emp.hiredate%TYPE 
 ); 
 
 TYPE nested_emp_type IS TABLE OF emp_rec_type; 
 
 emp_tab nested_emp_type; 
-- v_limit PLS_INTEGER := 3; 
-- v_counter PLS_INTEGER := 0; 
BEGIN 
 DELETE FROM emp WHERE deptno = 20 
 RETURNING empno, ename, hiredate -->使用returning 返回这几个列 
 BULK COLLECT INTO emp_tab; -->将前面返回的列的数据批量插入到集合变量 
 
 DBMS_OUTPUT.put_line( 'Deleted ' || SQL%ROWCOUNT || ' rows.' ); 
 COMMIT; 
 
 IF emp_tab.COUNT > 0 THEN -->当集合变量不为空时,输出所有被删除的元素 
 FOR i IN emp_tab.FIRST .. emp_tab.LAST 
 LOOP 
 DBMS_OUTPUT. 
 put_line( 
 'Current record ' 
 || emp_tab( i ).empno 
 || CHR( 9 ) 
 || emp_tab( i ).ename 
 || CHR( 9 ) 
 || emp_tab( i ).hiredate 
 || ' has been deleted' ); 
 END LOOP; 
 END IF; 
END; 
/

篇幅有限,关于bulk collect批量绑定方面的内容就介绍到这了,大家可以试着对存储过程中loop循环中的DML语句做适当改写,看是不是效率上有一定提升。实际上最好的应该是FORALL与BULK COLLECT结合使用,可以极大的提高执行效率。

后面会分享更多DBA方面内容,感兴趣的朋友可以关注下!

相关推荐

Linux基础知识之修改root用户密码

现象:Linux修改密码出现:Authenticationtokenmanipulationerror。故障解决办法:进入单用户,执行pwconv,再执行passwdroot。...

Linux如何修改远程访问端口

对于Linux服务器而言,其默认的远程访问端口为22。但是,出于安全方面的考虑,一般都会修改该端口。下面我来简答介绍一下如何修改Linux服务器默认的远程访问端口。对于默认端口而言,其相关的配置位于/...

如何批量更改文件的权限

如果你发觉一个目录结构下的大量文件权限(读、写、可执行)很乱时,可以执行以下两个命令批量修正:批量修改文件夹的权限chmod755-Rdir_name批量修改文件的权限finddir_nam...

CentOS「linux」学习笔记10:修改文件和目录权限

?linux基础操作:主要介绍了修改文件和目录的权限及chown和chgrp高级用法6.chmod修改权限1:字母方式[修改文件或目录的权限]u代表所属者,g代表所属组,o代表其他组的用户,a代表所有...

Linux下更改串口的权限

问题描述我在Ubuntu中使用ArduinoIDE,并且遇到串口问题。它过去一直有效,但由于可能不必要的原因,我觉得有必要将一些文件的所有权从root所有权更改为我的用户所有权。...

Linux chown命令:修改文件和目录的所有者和所属组

chown命令,可以认为是"changeowner"的缩写,主要用于修改文件(或目录)的所有者,除此之外,这个命令也可以修改文件(或目录)的所属组。当只需要修改所有者时,可使用...

chmod修改文件夹及子目录权限的方法

chmod修改文件夹及子目录权限的方法打开终端进入你需要修改的目录然后执行下面这条命令chmod777*-R全部子目录及文件权限改为777查看linux文件的权限:ls-l文件名称查看li...

Android 修改隐藏设置项权限

在Android系统中,修改某些隐藏设置项或权限通常涉及到系统级别的操作,尤其是针对非标准的、未在常规用户界面显示的高级选项。这些隐藏设置往往与隐私保护、安全相关的特殊功能有关,或者涉及开发者选项、权...

完蛋了!我不小心把Linux所有的文件权限修改了!在线等修复!

最近一个客户在群里说他一不小心把某台业务服务器的根目录权限给改了,本来想修改当前目录,结果执行成了根目录。...

linux改变安全性设置-改变所属关系

CentOS7.3学习笔记总结(五十八)-改变安全性设置-改变所属关系在以前的文章里,我介绍过linux文件权限,感兴趣的朋友可以关注我,阅读一下这篇文章。这里我们不在做过的介绍,注重介绍改变文件或者...

Python基础到实战一飞冲天(一)--linux基础(七)修改权限chmod

#07_Python基础到实战一飞冲天(一)--linux基础(七)--修改权限chmod-root-groupadd-groupdel-chgrp-username-passwd...

linux更改用户权限为root权限方法大全

背景在使用linux系统时,经常会遇到需要修改用户权限为root权限。通过修改用户所属群组groupid为root,此操作只能使普通用户实现享有部分root权限,普通用户仍不能像root用户一样享有超...

怎么用ip命令在linux中添加路由表项?

在Linux中添加路由表项,可以使用ip命令的route子命令。添加路由表项的基本语法如下:sudoiprouteadd<network>via<gateway>这...

Linux配置网络

1、网卡名配置相关文件回到顶部网卡名命名规则文件:/etc/udev/rules.d/70-persistent-net.rules#PCIdevice0x8086:0x100f(e1000)...

Linux系列---网络配置文件

1.网卡配置文件在/etc/sysconfig/network-scripts/下:[root@oldboynetwork-scripts]#ls/etc/sysconfig/network-s...

取消回复欢迎 发表评论: