在SQL Server 2005数据库中进行错误捕捉

作者:Blithe 时间:2008-12-02 14:39:00 

在SQL Server数据库中,如果执行Transact-SQL时出现了错误,我们可以使用两种捕捉错误的方法解决此问题,一种是在客户端代码(如c#、delphi等)中使用类似try...catch的语句进行捕捉;另外一种就是在Transact-SQL中利用Transact-SQL本身提供的错误捕捉机制进行捕捉。如果是因为Transact-SQL语句的执行而产生的错误,如键值冲突,使用第一种和第二种方法都可以捕捉,但是如果是逻辑错误,使用客户端代码进行捕捉就不太方便了。因此,本文针对如何使用Transact-SQL进行错误捕捉进行了深入的探讨。

一、非致命错误(non-fatal error)的捕捉

通过执行Transact-SQL而产生的错误可分为两种:致命错误(fatal error)和非致命错误(non-fatal error)。在Transact-SQL中只可以捕捉非致命错误(如键值冲突),而无法捕捉致命错误(如语法错误)。在Transact-SQL中可以通过系统变量@@ERROR判断最近执行的一条语句是否成功执行。如果发生了错误,@@Error的值大于0,否则值为0。下面举一个例子说明@@ERROR的使用。

假设有一个表table1,在这个表中有两个字段f1,f2。其中f1是主键。



INSERT INTO table1 VALUES(1, 'aa')
INSERT INTO table1 VALUES(1, 'bb')  --这条语句将产生一个错误
IF @@ERROR > 0
PRINT '键值冲突'


当执行第二条语句时发生键值冲突错误,@@ERROR被赋为错误号2627,因此输出结果显示'键值冲突'。使用@@ERROR系统变量时需要注意,@@ERROR只记录最近一次执行的Transact-SQL语句所发生的错误,如果最近一次执行的Transact-SQL没有发生错误,@@ERROR的值为0。因此,只能在被捕捉的那条Transact-SQL语句后使用@@ERROR。

在SQL Server中,不仅可以捕捉系统提供的错误,还可以自定义错误。有两种方法可以定义错误信息。

1. 使用sp_addmessage系统存储过程添加错误信息,然后使用RAISERROR抛出错误

sp_addmessage将错误号,错误级别、错误描述等信息添加到系统表中,然后使用RAISERROR根据相应的错误号抛出错误信息。用户自定义的信息应该从50001开始。



EXEC sp_addmessage @msgnum = 50001, @severity = 16,
@msgtext = 'sql encounter an error(%s).',
@lang = 'us_english'
EXEC sp_addmessage @msgnum = 50001, @severity = 16,
@msgtext = 'sql遇到了一个错误(%1!).'

如果使用的SQL Server版本是非英语版本,在添加本地错误信息时必须首先添加英文的错误信息。错误描述可以象c语言中的printf的格式字符串一样使用参数,如%s、%d。但要注意的是在英文版的错误信息中要使用%s、%d等形式,而在本地化的错误信息中要使用%1!、%2!等形式,在每个%?(1 <= ? <= n)后需要加一个!,而且%?的数目必须和英文版的错误信息的参数一致。

在未插入本地化错误信息时,RAISERROR将使用英文版的错误信息。当插入本地化错误信息时,RAISERROR使用本地化的错误信息。

RAISERROR(50001, 16, 1, '测试')

输出的结果:

服务器: 消息 50001,级别 16,状态 1,行 1

sql遇到了一个错误(测试).

其中'测试'字符串通过%1传入本地化的错误描述字符串中。

2.直接使用RAISERROR将错误抛出

使用第一种方法虽然使Transact-SQL语句看上去更整洁(这种方法类似于在编程语言中使用常量定义错误信息,然后在不同的地方通过错误编号引用这些错误信息。),但是这样做却使错误信息和数据库的耦合度增加,因为如果将这些带有RAISERROR的Transact-SQL放到别的SQL Server数据库上执行,由于在其它的数据库中还未添加错误信息,因此会产生RAISERROR调用错误,除非使用sp_addmessage将所需的错误信息再加入到其它的数据库中。

基于上述原因,RAISERROR不仅可以根据错误代码抛出错误信息,也可以直接通过错误描述格式字符串抛出错误信息。

RAISERROR('sql遇到了一个错误(%s)', 16, 1, '测试')

标签:
0
投稿

猜你喜欢

  • WEB手绘稿常用元素 之 箭头

    2009-05-08 12:33:00
  • 解析SQL Server与ASP互操作的时间处理

    2009-02-01 16:40:00
  • 古老的问题:清除浮动

    2009-02-12 11:21:00
  • asp如何用数据库制作一个多用户版的计数器?

    2010-06-16 09:51:00
  • JavaScript控制flash操作 兼容IE FF[译]

    2009-11-29 16:28:00
  • 懒懒交流会:ClassName的长命名 VS. 短命名

    2009-11-28 16:08:00
  • 如何获得上一个月份是几月?

    2009-11-23 20:38:00
  • Oracle9i取得建表和索引的DDL语句

    2010-07-20 12:59:00
  • ASP下标越界错误的解决方法

    2008-10-19 17:39:00
  • 将Reporting services的RDL文件拷贝到另外一台机器时报Data at the root level is invalid的解决方法

    2012-07-11 15:33:45
  • 语义、标准和样式

    2008-06-05 12:52:00
  • 如何优化JavaScript脚本的性能

    2008-02-19 21:47:00
  • IE地址栏显示网站图标制作方法

    2007-10-13 11:08:00
  • 历数Firefox2.0对XML处理的改进

    2007-11-27 12:41:00
  • mysql rand(x)生产重复随机序列

    2010-12-03 16:01:00
  • Linux下C连接MySQL出现错误解决一例

    2008-12-29 13:17:00
  • 是在客户端确认还是在服务器端确认?

    2010-07-14 21:05:00
  • asp 快钱网关接口 支付宝接口 财付通接口代码

    2011-03-08 10:55:00
  • 公用样式模板的设计制作

    2009-09-13 21:27:00
  • 可能被你忽略的 JavaScript 代码陷阱

    2009-12-26 18:16:00
  • asp之家 网络编程 m.aspxhome.com