一直以来有一个困惑,一直没解决,昨天一哥们问我这个问题,决心弄清楚,终于得到了答案。
先看下面这个函数:

create or replace function fn_test(c_xm varchar) return varchar2 as
V_P1 varchar(10);
begin
select name into V_p1 from t1 where 1 = 2;--将name查出赋值给v_p1
return 'test' || c_xm;
end;

这个函数很简单,是我写的一个测试函数,没什么意义,“select name into V_p1 from t1 where 1 = 2;”这句话有经验的人一看就知道它会报错,因为这个查询返回的结果集是空,会报一个错,将其赋值时,pl/sql引擎会认为它没有数据,是一个null,这很类似于java中的空指针异常。当我们调试该函数的时候,到这一句,立刻会报ORA-1403错误:没有数据。
但是如果在sql中调用该函数呢?执行以下查询:
select fn_test('1') from dual;
结果是返回一个空记录,没有任何报错。这是为什么呢?难道遇到了bug?如果是存储过程呢?无论如何调试还是直接调用,此处都会报错,有兴趣的可以验证一下,我就不验证了,因为我之前碰到过许多次了,所以一般在select into时,如果没有把握这个结果集一定有,都会select count一下然后再into。
这究竟是怎么回事呢?
下面是我的猜测:
对于查不到结果集来说,这不是什么很严重的错误,没有就没有了,不用报错吧?如我们执行一条sql,select * from t1 where 1=2;如果这个sql没有查到数据,难道就非得报个错?基于这个考虑,在sql调用函数时,如果这种NO__DATA_FOUND的异常,可能sql解析器就直接处理了,不用再报错了.因为它并不像诸如找不到表、找不到字段、没有权限等等的错误严重。
这仅仅是猜测,但是究竟为什么呢?
最后我在asktom的网站上找到了答案:

http://asktom.oracle.com/pls/asktom/f?p=100:11:0::::p11_question_id:10321465390114

从1楼这哥们的提问来看,他也是很郁闷,他甚至和我提出了一样的疑问:“为什么过程就没有这种问题?”
我看完了后,总结了一下,原因大致为:
当select a into b 时,如果没有命中结果集,则当前的查询为空,在sql语句的查询中,解析器仅仅是认为“没有数据(no data found)”而已,而不是将它作为一个错误,然后返回一个null,函数就此停止不再往下执行,tom的解释也比较诙谐:

Under the covers, SQL is raising back to the client application "hey buddy -- no_data_found".  The
client in this case says "ah hah, no data found means 'end of data'" and stops.

但是在pl/sql中却不是,pl/sql的处理方式却是将它认为是一个错误,

Under the covers, PLSQL is raising back to the client application "hey -- no_data_found.  The
client in this case says "uh-oh, wasn't expecting that from PLSQL -- sql sure, but not PLSQL. Lets
print out the text that goes with this exceptional condition and continue on"

NO_DATA_FOUND并不是一个错误,而且一个意外的情况,这类似于空指针异常,而这个意外情况只是没找到数据而已,当调用者不同时,对其的处理也不同,当sql查询调用时,遇到这个异常就认为是没有数据,然后返回一个Null,但是当PL/sql调用时,会认为这是一个不好的情况,转由异常处理块处理。
归根结底一句话,NO_DATA_FOUND都会由调用者捕获,只是调用者对这个异常的处理方式不一样而已。如果想在sql调用时报错怎么办?其实很简单,捕获这个NO_DATA_FOUND异常,然后raise即可:

create or replace function fn_test(c_xm varchar) return varchar2 as
V_P1 varchar(10);
begin
select name into V_p1 from t1 where 1 = 2;
return 'test' || c_xm;
exception
when no_data_found then
/*RAISE_APPLICATION_ERROR(-20000, 'no data found');*/--抛出自定义的异常也行
raise program_error;
end;

这样,当再执行到该句时,立刻转到异常处理块,抛出一个非NO_DATA_FOUND异常,调用者不认识,认为是一个错误或者很严重的异常,只能报错给客户端了。

对于【异常总是会抛出,只是客户端(调用者)对其处理方式不一样】,可以这样理解:
当用pl/sql调试时,运行到1403异常处,pl/sql调试器的处理方式就是立刻弹出一个错误信息;而sql调用时,这地方异常也会抛出,但是sql查询器会认为,哦,没有数据,查询器选择了用一个null值应对这个异常,而作为执行sql的我们,所看到的就是一个空值,而没有报错。
这类似于我们写java程序对异常的处理,有的异常我们会直接抛给用户,让用户知道出错了,而有的异常被我们吃掉,然后选择了别的处理方法,用户看到的是另外一个情形,他根本不知道后台有异常发生。
这也就是对于【异常存在,只是怎么应对】的解释。

有个结论:如果在function中,如果某行报了NO_DATA_FOUND,也没有处理块,那么不好意思,pl/sql语句就此就不在执行,这和普通的java程序是一样的,什么地方抛出异常,程序在此就停止运行,要么转到异常处理部分,要么就此stop,如果在sql查询语句中调用这个fn_test函数:
select fn_test('1') from dual;
执行函数调用的过程用伪代码表示如下:
begin:
select fn_test('') from dual;--开始解析sql查询语句
call fn_test;--发现值来自于函数,开始调用fn_test
var result;--定义临时变量接收结果
try{
result=fn_test('');
}catch(NO_DATA_FOUND){--如果是NO_DATA_FOUND异常则null处理
result=null;
}catch(OTHERS){--如果其他异常则抛出
throw others;
}
select result from dual;
end;

以上过程模拟了select语句调用函数的过程,如果出现了异常,在报异常的地方函数就此停止运行,不再往下执行。
验证:

create or replace function fn_test(c_xm varchar) return varchar2 as
V_P1 varchar(10);
begin
select name into V_p1 from t1 where 1 = 2;--NO_DATA_FOUND,也会报错,但是sql解析器会以null返回
select 1/0 into v_p1 from dual;--除数为0,会报错
return 'test' || c_xm;
end;

当再次执行selectu语句的时候,并没有报除数为0的错误,因为查询在第一条语句就停止了,不再往下执行,如果去掉第一条语句:

create or replace function fn_test(c_xm varchar) return varchar2 as
V_P1 varchar(10);
begin
--select name into V_p1 from t1 where 1 = 2;--NO_DATA_FOUND,注释掉该行
select 1/0 into v_p1 from dual;--除数为0,会报错
return 'test' || c_xm;
end;

执行查询,立刻报错:ORA-01476:除数为0。如下图:

当执行了异常处理时,若发生了异常,则会立即跳转到异常块中,这和java是一样的,可以选择捕获NO_DATA_FOUND异常然后外抛。

create or replace function fn_test(c_xm varchar) return varchar2 as
V_P1 varchar(10);
begin
select name into V_p1 from t1 where 1 = 2;--NO_DATA_FOUND,会立即跳转到exception块,不再继续执行
select 1 / 0 into v_p1 from dual; --除数为0,会报错,但是这句没有机会执行了
return 'test' || c_xm;
exception
when NO_DATA_FOUND then
raise_application_error('-20000', '没找到数据');--异常外抛给调用者,直接报错
end;

如下图:

也可以在异常中返回一个有意义的提示,告诉调用者一个有意义的信息,如:

create or replace function fn_test(c_xm varchar) return varchar2 as
V_P1 varchar(10);
begin
select name into V_p1 from t1 where 1 = 2;--NO_DATA_FOUND,会立即跳转到exception块,不再继续执行
select 1 / 0 into v_p1 from dual; --除数为0,会报错,但是这句没有机会执行了
return 'test' || c_xm;
exception
when NO_DATA_FOUND then
return '没有找到数据!';
end;

结果如下图:

这个结论适用于其他情况,无论是在loop中,还是单一查询,只要报了NO_DATA_FOUND异常,都会立即stop,要么跳转到exception,要么返回null,不再继续执行,其实原理很简单,和java是一样的,很好理解。

【原创】Oracle函数中对于NO_DATA_FOUND异常处理的研究的更多相关文章

  1. Oracle函数中对于NO_DATA_FOUND异常处理的研究

    一直以来有一个困惑,一直没解决,昨天一哥们问我这个问题,决心弄清楚,终于得到了答案.先看下面这个函数: create or replace function fn_test(c_xm varchar) ...

  2. oracle 函数中,一定要注意出现空记录和多条记录的处理方法

    今天折腾了3个小时,为一个以前不知道的oracle函数机制: 在sql查询中,如果一个查询未能获取记录,oracle不会报错 如select aa from bb where 1=2; 但在oracl ...

  3. Oracle 函数中动态执行语句

    函数: 1 create or replace function fn_test(tablename in varchar2) return number is sqls ); rtn ):; beg ...

  4. oracle函数中lead,lag,over,partition by 的使用

    lead,lag函数的分析 http://blog.csdn.net/mazongqiang/article/details/7621328 举例如下: SQL> select *  from ...

  5. 浅谈Oracle函数返回Table集合

    在调用Oracle函数时为了让PL/SQL 函数返回数据的多个行,必须通过返回一个 REF CURSOR 或一个数据集合来完成.REF CURSOR 的这种情况局限于可以从查询中选择的数据,而整个集合 ...

  6. Oracle 函数 “申请通过后,将该表中循环遍历到的所有内容插到另一个表中”

    create or replace function mcode_apply_insert_material(p_mca_no VARCHAR2, p_action VARCHAR2, p_wf_no ...

  7. Oracle 函数 “判断数据表中不存在的数据,才允许通过”

    create or replace function mca_detail_material_val(p_material_code VARCHAR2, --实参 p_material_name VA ...

  8. Oracle 函数 “把当前的用户(审核人,审核通过后)插入到数据表中”

    create or replace function mcode_apply_update_personnel(p_mca_no VARCHAR2, -- 参数(实参) p_action VARCHA ...

  9. Oracle数据库中调用Java类开发存储过程、函数的方法

    Oracle数据库中调用Java类开发存储过程.函数的方法 时间:2014年12月24日  浏览:5538次 oracle数据库的开发非常灵活,不仅支持最基本的SQL,而且还提供了独有的PL/SQL, ...

随机推荐

  1. 【M25】将构造方法和非成员方法虚化

    1.所谓虚化,就是根据引用或者指针的真实类型,决定调用哪个方法. 2.构造方法虚化,就是根据引用(或者指针)的真实类型,构造出一个对象,如果指针的真实类型是Base,返回Base*:如果指针的真实类型 ...

  2. 你应该知道的JavaScript中NaN的秘密

    NaN,不是一个数字,是一种特殊的值来代表不可表示的值,使用typeof或其他任何与之比较的处理方式,‘NaN’则会引起一些混乱, 一些操作会导致NaN值的产生.这里有些例子: Math.sqrt(- ...

  3. Codeforces Round #181 (Div. 2) C. Beautiful Numbers 排列组合 暴力

    C. Beautiful Numbers 题目连接: http://www.codeforces.com/contest/300/problem/C Description Vitaly is a v ...

  4. codeforces Gym 100187F F - Doomsday 区间覆盖贪心

    F. Doomsday Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100187/problem/F ...

  5. Codeforces Round #189 (Div. 1) B. Psychos in a Line 单调队列

    B. Psychos in a Line Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/problemset/p ...

  6. TZC 1472 逆置正整数,去前导零 (java一句话秒杀)

    逆置正整数 http://acm.tzc.edu.cn/acmhome/problemdetail.do?&method=showdetail&id=1472 时间限制(普通/Java ...

  7. 因为改 UOM conversion 导致库存数量和財务上的数据错误

    轻易改变 UOM conversion 会导致库存数量混乱, 也会造成財务上的数据错误. 我们这里做一个 case 来详细分析一下. 1. 開始 Carton 和 Each 的比例是 1 : 1. 2 ...

  8. Ubuntu:Target filesystem doesn't have /sbin/init (Slax 解决)

    计算机(Ubuntu)因为异常断电或是其它原因,再次启动时.非常不幸的出现: Killed mount: mounting /dev on /root/dev failed: No such file ...

  9. iOS开发——UI篇Swift篇&UITextView

    UITextView 一:UITextView使用及其属性的设置 titleLabel.text = titleString //创建UITextView对象 textView = UITextVie ...

  10. Quartz.Net实现定时任务调度

    Quartz.Net介绍: Quartz一个开源的作业调度框架,OpenSymphony的开源项目.Quartz.Net 是Quartz的C#移植版本. 它一些很好的特性: 1:支持集群,作业分组,作 ...