在未插入本地化错误信息时,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, \'测试\')
二、逻辑错误的捕捉
在实际应用中,更多的是由于某些业务要求而产生的逻辑错误。这些错误无法通过@@ERROR进行捕捉。如果使用客户端代码进行捕捉,那么Transact-SQL必须一条一条地执行。如果使用存储过程,那么发生在存储过程内部的逻辑错误就很难在客户端代码中进行捕捉,因此,下面将讨论如何使用Transact-SQL捕捉逻辑错误。
所谓逻辑错误,就是在执行完Transact-SQL后,执行结果与业务要求的结果不符而产生的。为了说明如何处理逻辑错误,我们再建立一个表table2,这个表的结构和table1完全一样,只是f1字段不再是主键了。然后建立一个存储过程,它的功能是在table1和table2中同时插入一条记录,但是这条记录必须满足两个条件。
1、f1值不能大于100。
2、要插入的记录在table1中不存在,如果存在,在table1和table2中都不插入这条记录。
CREATE PROCEDURE p1(@Num int)ASDECLARE @Error int, @RowCount intBEGIN TRANSACTIONINSERT INTO table2 VALUES(@Num, \'p\')IF @Num > 100BEGINRAISERROR(\'%s的值不能大于100。\', 16, 1, \'@Num\') ROLLBACK TRANSACTIONRETURN 1ENDELSEBEGINSELECT f1 FROM table1 WHERE f1 = @NumIF @@ROWCOUNT > 0BEGINRAISERROR(\'table1中已经存在%d了。\', 16, 1, @Num)ROLLBACK TRANSACTIONRETURN 2ENDELSEBEGININSERT INTO table1 VALUES(@Num, \'p\')COMMIT TRANSACTIONRETURN 0ENDEND
在这个存储过程中一开始使用BEGIN TRANSACTION显示地开始一个事务,然后当上述两种错误发生时使用ROLLBACK TRANSACTION恢复到初始状态,如果成功插入,使用COMMIT TRANSACTION提交改变。可以通过如下语句进行调用。
DECLARE @ErrNum int
EXEC @ErrNUm = p1 2
PRINT @ErrNum
可以通过@ErrNum得到p1返回的错误代码,如果返回0,表示执行成功。
sql Server2005中错误捕捉的新功能
虽然在以前的SQL Server版本中可以通过一些技巧实现错误捕捉,但有时需要增加一些额外的开销,如在p1中使用了SELECT语句。庆幸的是在SQL Server2005中提供了和大多数编程语言类似的try...catch错误捕捉功能,从而使Transact-SQL第一次可以真正地进行错误捕捉。使用try...catch可以将p1的下半部分改写为如下形式。