?
下面通过一个简单的存储过程代码段来说明问题:
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才引入的一个函数,意为找到出现错误的行。
?
?
?
?