日期:2014-05-16  浏览次数:20546 次

oracle存储过程异常处理及时间戳变量类型

?

下面通过一个简单的存储过程代码段来说明问题:

CREATE OR REPLACE PROCEDURE load_error IS

  V_td1         test1.td1%TYPE;
  v_td1xh       test1.td1xh%TYPE;
  v_gxsj         timestamp ; -- 时间戳变量
  v_zhxgsj       timestamp ;
  V_COUNT NUMBER ; --用于临时存放查询结果数量,如果数量大于2时,以便循环取出处理
  V_STR VARCHAR(100) ; --用于存放临时变量
  v_errorCode load_ajxx_error.errorcode%TYPE ;
  v_errorText load_ajxx_error.errortext%TYPE ;
  v_errorline load_ajxx_error.errorline%TYPE ;

  -- 定义游标类型(用做动态游标)
  TYPE CURSOR_TYPE IS REF CURSOR;   
    CURSOR_DYNAMIC CURSOR_TYPE; 

  CURSOR test2_CURSOR IS
    SELECT td2xh,td21,td22,td23,td24 FROM test2 ;

BEGIN
  -- 循环变更游标
  FOR test2_RECORD IN test2_CURSOR LOOP 
    BEGIN 
      v_td1xh := test2_RECORD.td2xh ;
      V_td1 := test2_RECORD.td21 ;   -- 此处可能报缓冲区不足的错误
      --dbms_output.put_line(v_zhxgsj || ',' || v_gxsj) ;
    -- 截获抛出的异常
    EXCEPTION 
      WHEN OTHERS THEN
        v_errorcode := SQLCODE ; -- 异常代码
        v_errorText := Sqlerrm ; -- 异常代码描述
        v_errorline := dbms_utility.format_error_backtrace() ;  -- 异常所在行(只在oracle 10g才能使用)
        INSERT INTO load_ajxx_error(xh,errorline, errorcode, errortext, tbsj) VALUES(v_td1xh,v_errorline,v_errorcode,v_errorText,SYSDATE) ;
    END ;
    COMMIT;
  END LOOP ;
END load_error;

?

1、时间戳变更介绍:

?????? 在用存储过程导数据时,一般都要用时间字段(时间戳)来判断是否需要update。但是oracle中的如果把存储过程的变量定义成DATE的,那么oracle会自动将数据表中的时间字段转换为日期型,而把时间信息截取掉了。但是我们要导入的数据一天可能会更新多次,而且实时性要求较高,故,必须精确到秒。此时就用到了oracle的timestamp变量,此变量产生的值为:15-3月 -14 12.31.59.000000 上午,这个日期精确足够我们使用了。

?

2、oarcle异常处理介绍:

?????? 在使用存储过程导数据时,异常处理是必不可少的,因为我们需要知道在哪些操作上发生了错误,造成数据没有正确插入。如果只是某一条数据出现了问题,那么我们还要继续处理其他没有问题的数据,待所有数据都处理完成后,我们需要到日志表里面查看一下,到底是哪些数据出了问题,出了什么问题,再经过分析解决掉出现问题的数据。

具体如下:

???????一般我们要用游标对一组数据进行处理,所以我们就要在游标的循环体中加入一个代码块,来处理整个游标周期中可能出现的问题:

BEGIN
  -- 循环变更游标
  FOR test2_RECORD IN test2_CURSOR LOOP 
    BEGIN 
      v_td1xh := test2_RECORD.td2xh ;
      V_td1 := test2_RECORD.td21 ;   -- 此处可能报缓冲区不足的错误
      --dbms_output.put_line(v_zhxgsj || ',' || v_gxsj) ;
    -- 截获抛出的异常
    EXCEPTION 
      WHEN OTHERS THEN
        v_errorcode := SQLCODE ; -- 异常代码
        v_errorText := Sqlerrm ; -- 异常代码描述
        v_errorline := dbms_utility.format_error_backtrace() ;  -- 异常所在行(只在oracle 10g才能使用)
        INSERT INTO load_ajxx_error(xh,errorline, errorcode, errortext, tbsj) VALUES(v_td1xh,v_errorline,v_errorcode,v_errorText,SYSDATE) ;
    END ;
    COMMIT;
  END LOOP ;
END ;

?FOR循环中的begin ... exception ... end ;是必不可少的。

?

SQLCODE :oracle中的关键字,意思为错误代码。

SQLERRM :oracle中的关键字,意思为错误描述。有的时候可能会很长,可以用substr截取一下。一般截取前200位就可以,我们只需要了解一些简单的信息即可 。

dbms_utility.format_error_backtrace() :在oracle 10g才引入的一个函数,意为找到出现错误的行。

?

?

?

?

1 楼 xiaojie921 3 小时前  
aaaaaa