分类: SQL/PLSQL 基础2010-12-22 16:23 4084人阅读 评论(0) 收藏 举报

--===============================

-- PL/SQL -->隐式游标(SQL%FOUND)

--===============================

在PL/SQL中,游标的使用分为两种,一种是显示游标,一种是隐式游标,显示游标的使用需要事先使用declare来进行声明,其过程包括

声明游标,打开游标,从游标提取数据,关闭游标。该方式多用于处理select语句返回的多行数据的情形。而隐式游标则由则由系统自动定义

,当DML被使用时,Oracle为每一个不属于显示游标的DML语句都创建一个隐式游标,其声明、打开、关闭都是系统自动进行。多用于配合DML

返回单行数据的处理。

有关显示游标的使用,请参考:PL/SQL --> 游标

一、隐式游标的定义及其属性

定义

隐式游标则由则由系统自动定义,非显示定义游标的DML语句即被赋予隐式游标属性。其过程由oracle控制,完全自动化。隐式游标

的名称是SQL,不能对SQL游标显式地执行OPEN,FETCH,CLOSE语句。

属性

类似于显示游标,隐式游标同样具有四种属性,只不过隐式游标以SQL%开头,而显示游标以Cursor_name%开头

通过SQL%总是只能访问前一个DML操作或单行SELECT操作的游标属性,用于判断DML执行的状态和结果,进而控制程序的流程

SQL%ISOPEN

游标是否打开。当执行select into ,insert update,delete时,Oracle会隐含地打开游标,且在该语句执行完毕或隐含地关闭

游标,因为是隐式游标,故SQL%ISOPEN总是false

SQL%FOUND

判断SQL语句是否成功执行。当有作用行时则成功执行为true,否则为false。

SQL%NOTFOUND

判断SQL语句是否成功执行。当有作用行时否其值为false,否则其值为true。

SQL%ROWCOUNT

在执行任何DML语句之前,SQL%ROWCOUNT的值都是NULL,对于SELECT INTO语句,如果执行成功,SQL%ROWCOUNT的值为,如果没有

成功,SQL%ROWCOUNT的值为,同时产生一个异常NO_DATA_FOUND。

二、演示

1.SQL%FOUND的使用

DECLARE

v_empno emp.empno%TYPE:=&no;

BEGIN

UPDATE emp SET sal=sal+200     --根据给定的empno,更新一条记录

WHERE empno=v_empno;

IF SQL%FOUND THEN              --使用SQL游标属性SQL%FOUND作为判断条件

COMMIT;

DBMS_OUTPUT.PUT_LINE('SQL code is executed successful');

ELSE

DBMS_OUTPUT.PUT_LINE('The Employee is not exist');

ROLLBACK;

END IF;

END;

Enter value for no: 7788

old   2:   v_empno emp.empno%TYPE:=&no;

new   2:   v_empno emp.empno%TYPE:=7788;

SQL code is executed successful

PL/SQL procedure successfully completed

2.SQL游标的综合应用(根据SQL游标的不同属性返回不同的结果)

DECLARE

v_dept emp.deptno%TYPE := &no;

BEGIN

IF SQL%ROWCOUNT >= 0 THEN  --判断更新前SQL%ROWCOUNT的属性

DBMS_OUTPUT.PUT_LINE('SQL%ROWCOUNT value is ' || SQL%ROWCOUNT ||

'before updated');

ELSE

DBMS_OUTPUT.PUT_LINE('SQL%ROWCOUNT value is NULL before updated');

END IF;

UPDATE emp SET sal = sal + 200 WHERE deptno = v_dept;

IF SQL%FOUND THEN    --判断SQL%FOUND的属性

DBMS_OUTPUT.PUT_LINE('SQL code is executed successful');

DBMS_OUTPUT.PUT_LINE('SQL%Found is TRUE');

ELSE

DBMS_OUTPUT.PUT_LINE('No such department');

DBMS_OUTPUT.PUT_LINE('SQL%Found is FALSE');

END IF;

IF SQL%NOTFOUND THEN    --判断SQL%NOTFOUND的属性

DBMS_OUTPUT.PUT_LINE('SQL%NotFound is TRUE');

ELSE

DBMS_OUTPUT.PUT_LINE('SQL%NotFound is FALSE');

END IF;

IF SQL%ISOPEN THEN    --判断SQL%ISOPEN的属性

DBMS_OUTPUT.PUT_LINE('SQL%ISOPEN is TRUE');

ELSE

DBMS_OUTPUT.PUT_LINE('SQL%ISOPEN is FALSE');

END IF;

DBMS_OUTPUT.PUT_LINE('The rows updated is :' || SQL%ROWCOUNT ||

' rows by SQL Cursor'); --判断SQL%ROWCOUNT的属性

END;

Enter value for no: 10   --下面是成功更新后的结果

SQL%ROWCOUNT value is NULL before updated

SQL code is executed successful

SQL%Found is TRUE

SQL%NotFound is FALSE

SQL%ISOPEN is FALSE

The rows updated is :3 rows by SQL Cursor

Enter value for no: 80   --下面是未成功更新后的结果

SQL%ROWCOUNT value is NULL before updated

No such department

SQL%Found is FALSE

SQL%NotFound is TRUE

SQL%ISOPEN is FALSE

The rows updated is :0 rows by SQL Cursor

3.SELECT INTO时,隐式游标的使用

SELECT INTO用于将单行结果集放置到变量之中。

SELECT INTO处理的结果包括两种种情况

查询结果返回单行,SELECT INTO被成功执行

查询结果没有返回行,PL/SQL将抛出no_data_found异常

查询结果返回多行,PL/SQL将抛出too_many_rows 异常

对于上述两种异常发生时,类似于普通异常处理,程序控制权转移到异常处理部分(如没有异常处理则程序中断)。对于异常被激后发

,SQL游标的四个属性在此将不可使用,如下面的例子:

DECLARE

v_ename emp.ename%TYPE;

BEGIN

SELECT ename INTO v_ename FROM emp WHERE empno=&no;

IF  SQL%ROWCOUNT=0 OR SQL%NOTFOUND THEN

DBMS_OUTPUT.PUT_LINE('The record '||&no||' is not exist!');

ELSE

DBMS_OUTPUT.PUT_LINE('The name for record '||&no||' is '||v_ename );

END IF;

EXCEPTION

WHEN NO_DATA_FOUND THEN

DBMS_OUTPUT.PUT_LINE('No data found for '||&no);

END;

Enter value for no:70

No data found for 70

Enter value for no:7788

The name for record 7788 is SCOTT

从上面的演示中可以看到,当select into没有返回行时,IF  SQL%ROWCOUNT=0 OR SQL%NOTFOUND THEN 语句并没有被执行。

使用下面改进过的代码来执行,即可以将SQL游标属性判断放置到EXCEPTION部分

DECLARE

v_ename emp.ename%TYPE;

BEGIN

SELECT ename INTO v_ename FROM emp WHERE empno=&no;

IF SQL%NOTFOUND THEN

DBMS_OUTPUT.PUT_LINE('The record '||&no||' is not exist!');

ELSE

DBMS_OUTPUT.PUT_LINE('The name for record '||&no||' is '||v_ename );

END IF;

EXCEPTION

WHEN NO_DATA_FOUND THEN

IF SQL%NOTFOUND THEN

DBMS_OUTPUT.PUT_LINE('The record '||&no||' is not exist!');

DBMS_OUTPUT.PUT_LINE('No data found for '||&no);

ELSE

DBMS_OUTPUT.PUT_LINE('The name for record '||&no||' is '||v_ename );

END IF;

END;

Enter value for no:80

The record 80 is not exist!

No data found for 80

更多关于隐式游标的探讨,请参考:IMPLICIT CURSOR ATTRIBUTE SQL%NOTFOUND NOT WORKING

三、更多参考

有关SQL请参考

SQL 基础--> 子查询

SQL 基础-->多表查询

SQL基础-->分组与分组函数

SQL 基础-->常用函数

SQL 基础--> ROLLUP与CUBE运算符实现数据汇总

SQL基础-->层次化查询(START BY ... CONNECT BY PRIOR)

有关PL/SQL请参考

PL/SQL --> 语言基础

PL/SQL --> 流程控制

PL/SQL --> 存储过程

PL/SQL --> 函数

PL/SQL --> 游标

PL/SQL -->隐式游标(SQL%FOUND)

PL/SQL --> 异常处理(Exception)

PL/SQL --> PL/SQL记录

PL/SQL --> 包的创建与管理

PL/SQL --> 包重载、初始化

PL/SQL --> DBMS_DDL包的使用

PL/SQL --> DML 触发器

PL/SQL --> INSTEAD OF 触发器

PL/SQL -->隐式游标(SQL%FOUND)的更多相关文章

  1. PL/SQL — 隐式游标

    一.隐式游标的定义及其属性 定义 隐式游标由系统自动定义,非显示定义游标的DML语句即被赋予隐式游标属性.其过程由oracle控制,完全自动化.隐式游标的名称是SQL,不能对SQL游标显式地执行OPE ...

  2. 【Oracle】PL/SQL 显式游标、隐式游标、动态游标

    在PL/SQL块中执行SELECT.INSERT.DELETE和UPDATE语句时,Oracle会在内存中为其分配上下文区(Context Area),即缓冲区.游标是指向该区的一个指针,或是命名一个 ...

  3. Oracle pl/sql 显示游标和隐式游标

    显示游标 一.定义语法:        CURSOR <游标名> IS         <SELECT 语句>         [FOR UPDATE | FOR UPDATE ...

  4. PL/SQL — 显式游标

    一.游标的相关概念及特性 1.定义 通过游标方式定位到结果集中某个特定的行,然后根据业务需求对该行进行相应特定的操作. 2.分类 显示游标: 用户自定义游标,用于处理select语句返回的多行数据. ...

  5. oracle的隐式游标

    游标的概念:     游标是SQL的一个内存工作区,由系统或用户以变量的形式定义.游标的作用就是用于临时存储从数据库中提取的数据块.在某些情况下,需要把数据从存放在磁盘的表中调到计算机内存中进行处理, ...

  6. Oracle异常处理内容,隐式游标

    异常处理 create or replace procedure pr_test3(v_bh in varchar2,v_xx out t_hq_ryxx%rowtype) is begin sele ...

  7. Oracle游标cursor1基础和隐式游标

    --指向表行的指针,一次一行,一般向前移动 Resultset --游标永远代代表的是一行数据. /* 使用步骤 第一步:声明游标,就像是声明一个变量样. 游标的关键字就是cursor. Declar ...

  8. ORACLE的显式游标与隐式游标

    1)查询返回单行记录时→隐式游标: 2)查询返回多行记录并逐行进行处理时→显式游标 显式游标例子: DECLARE CURSOR CUR_EMP IS SELECT * FROM EMP; ROW_E ...

  9. Oracle 隐式游标 存储过程

    --隐式游标 注意变量赋值用(:=) 连接符用(||)而不是加号(+) DECLARE v_pk T_PLAT_KEYWORD.ID%TYPE; --主键 v_amount_message T_PLA ...

随机推荐

  1. 「 Luogu P2657 」 windy数

    # 题目大意 给出区间 $[a,b]$,求出区间中有多少数满足下列两个条件 不含有前导 $0$. 相邻两个数字之差的绝对值至少是 $2$. # 解题思路 数位 $DP$,用记忆化搜索来实现.设 $dp ...

  2. [模板] Treap

    插入x 删除x 查询排名为x的数 查询x的排名 求x的前驱.后继 //Stay foolish,stay hungry,stay young,stay simple #include<iostr ...

  3. 主席树模板poj 2104

    资料1:http://blog.csdn.net/regina8023/article/details/41910615 资料2:模板来源:http://www.cnblogs.com/lidaxin ...

  4. c++中的三角函数

    c++中想求cos或sin: 1.首先得包含头文件,include<math.h> 2.sin(),cos(),中是弧度数,即若是角度a,则应写成cou<<sin(a*pi/1 ...

  5. Swagger UI教程

    文档源地址 http://www.68idc.cn/help/makewebs/qitaasks/20160621620667.html Swagger-UI本身只提供在线测试功能,要集成它还需要告诉 ...

  6. CPLD和FPGA中不同电压的JTAG电路设计注意事项

    在初次的cpld电路设计的时候,遇到了这样的一个问题,整个系统是3.3V的系统,选用的cpld是XC9536-10VQ44C,芯片供电电压5V.他的io可以配置成3.3V和5V两种形式,因此,选用的时 ...

  7. POJ-3692Kindergarten,求最大独立集!

    Kindergarten Time Limit: 2000MS   Memory Limit: 65536K       Description In a kindergarten, there ar ...

  8. Apple & APPID & iOS & React Native

    Apple & APPID & iOS & React Native 在没有 苹果开发者账号证书 APPID, ios 是否支持导出 app https://developer ...

  9. 括号序列(Poj1141)

    Poj1141 题目描述: 定义合法的括号序列如下: 1 空序列是一个合法的序列 2 如果S是合法的序列,则(S)和[S]也是合法的序列 3 如果A和B是合法的序列,则AB也是合法的序列 例如:下面的 ...

  10. Tyvj 1221 微子危机——战略

    背景 №.3Summer联盟战前兵力战略转移. 描述 Summer的兵力分布在各个星球上,现在需要把他们全部转移到某个星球上.Summer一共拥有N个星球(1-N),你要把这N个星球上的兵力转到第M个 ...