【Oracle】异常信息的加工处理

引言

    很多时候,我们调用oracle存储过程都会发生各种各样的异常信息,例如ORA-12899值过大,ORA-01400不能插入空值等.虽然说这类异常是前端没控制到位的缘故,但是现实很难100%完全控制住,所以一旦发生了这类异常,并返回的前端显示时就会造成很差的用户体验了.或许,我们可以将这类异常加工一下,包装得好看些,让用户看得懂发生了什么事情,能够自行处理.下面是我的一点尝试.

    我们通常会在oracle存储过程的异常处理得到异常信息,如:

Exception
    When Others Then
     rollback;
     dbms_output.put_line(sqlerrm);
     Raise_Application_Error(-20000,sqlerrm); 

     然而,我们将会得到下面的异常信息,如

ORA-12899: 列 "NIS"."T_CZL_ACCOUNT"."VC_ACCOUNT" 的值太大 (实际值: 35, 最大值: 20)

     虽然说我们99%的程序员都看得懂这个是什么意思,但是也有99%的用户看不出个所以然.我们的目标是意外发生了,用户依然看得懂.接下来,我需要创建一个函数,专门处理这类的异常,返回大部分用户看得懂的信息.函数代码如下:

create or replace function f_czl_geterror(message in varchar2)
return varchar2 is
   Result varchar2(1000);
  
   num1 number:=0;
   num2 number:=0;
   num3 number:=0;
   num4 number:=0;
   num5 number:=0;
   num6 number:=0;
   num7 number:=0;
   num8 number:=0;
   num9 number:=0;
   num10 number:=0;
   str1 varchar2(1000);             
   str2 varchar2(1000);  
   str3 varchar2(1000);
   str4 varchar2(1000);
   str5 varchar2(1000);                                
   str6 varchar2(1000);             
   str7 varchar2(1000);  
   str8 varchar2(1000);
   str9 varchar2(1000);
   str10 varchar2(1000);
begin
   
    if instr(message, 'ORA-12899')>0 then
     num1:=instr(message,'ORA-12899'); --得到 ORA-12899的所在位置
     str1:=substr(message,num1);    --得到ORA-12899后面的所有字符 
     num2:=instr(str1,'"',1,3);    --得到ORA-12899后面第三个"的位置
     num3:=instr(str1,'"',1,4);    --得到ORA-12899后面第四个"的位置
     str2:=substr(str1,num2+1,num3-num2-1);  --得到表名
     num4:=instr(str1,'"',1,5);    --得到ORA-12899后面第五个"的位置
     num5:=instr(str1,'"',1,6);    --得到ORA-12899后面第六个"的位置
     str3:=substr(str1,num4+1,num5-num4-1);  --得到字段名
     
     num6:=instr(str1,':',1,2);    --得到ORA-12899后面第二个:的位置
     num7:=instr(str1,',',1,1);    --得到ORA-12899后面第1个,的位置
     str5:=substr(str1,num6+1,num7-num6-1);  --得到输入的长度
     
     num8:=instr(str1,':',1,3);    --得到ORA-12899后面第二个:的位置
     num9:=instr(str1,')',1,1);    --得到ORA-12899后面第1个,的位置
     str6:=substr(str1,num8+1,num9-num8-1);  --得到最大的长度
       
     --得到对应的字段注释
     select t.comments into str4 from SYS.USER_COL_COMMENTS t where t.column_name=str3 and t.table_name=str2;
     
     Result:=str4||'长度超出限制,最大字符数为'||str6||',您输入的字符数为'||str5||'.';
    elsif  instr(message, 'ORA-01400')>0  then
       
     num1:=instr(message,'ORA-01400'); --得到 ORA-01400的所在位置
     str1:=substr(message,num1);    --得到ORA-01400后面的所有字符 
     num2:=instr(str1,'"',1,3);    --得到ORA-01400后面第三个"的位置
     num3:=instr(str1,'"',1,4);    --得到ORA-01400后面第四个"的位置
     str2:=substr(str1,num2+1,num3-num2-1);  --得到表名
     num4:=instr(str1,'"',1,5);    --得到ORA-01400后面第五个"的位置
     num5:=instr(str1,'"',1,6);    --得到ORA-01400后面第六个"的位置
     str3:=substr(str1,num4+1,num5-num4-1);  --得到字段名
     
      --得到对应的字段注释
     select t.comments into str4 from SYS.USER_COL_COMMENTS t where t.column_name=str3 and t.table_name=str2;
     
      Result:=str4||'不能为空,请输入内容.';
   elsif  instr(message, 'ORA-01438')>0  then
       
  
      Result:=str4||'数字长度超出限制,请检查!';
    else
      
        Result:=message;
    end if;
    
      return(Result);
end f_czl_geterror;

 

    然后我们在存储过程的异常处理中用上这个函数,如:

Exception
    When Others Then
     rollback;
     dbms_output.put_line(f_czl_geterror(sqlerrm));
     Raise_Application_Error(-20000,f_czl_geterror(sqlerrm));   

    再测试看看,我们得到了:

账号长度超出限制,最大字符数为 20,您输入的字符数为 35.

    起码有的用户看得懂了是吧,咱们的目标算是达成了.

小结

  上文介绍了如何将oracle的系统异常信息转换成用户看得懂的信息.其实这只是一种补救的措施罢了,某种程度上是'欺骗'了用户,程序是正常运行的,但是不能否认的是我们的程序依然不完善.但是,有的补救好过没有是吧.最后,如果您有更好的建议,请不吝指教.

posted @ 2015-03-31 09:53  Caizl  阅读(226)  评论(0编辑  收藏  举报