谈谈基于SQL Server 的Exception Handling[上篇]

对于所有的开发人员来说,Exception Handling是我们每天都要面对的事情。对于基于Source CodeException Handling,我想大家已经司空见惯了,但是对于Database级别的Exception Handling,就没有那么常见了。在这篇文章中,我将会介绍我对于基于Database编程中Exception Handling的一些粗浅的认识:在编写Stored Procedure时,如何抛出一个可预知的ExceptionADO.NET如何处理从Database抛出的Exception,如何保存基于Database ExceptionError Message,如何在Database.NET Application之间进行消息的传递[注:这里的Database主要指SQL Server]

@@ERROR
RAISEERROR

TRY CATCH & Return

Error message & sys.messages

ADO.NET Exception Handling

SqlException

InfoMessage

一、    @@ERROR

@@ERROR是我们经常使用的系统函数,其返回类型为INT,用以表示上一个语句的执行是否遇到错误,0便是语句正常执行,非0则以为着某个错误的产生。比如下面的一个例子:我在SQL Server Management Studio中执行下面一段简单的SQL

《谈谈基于SQL Server 的Exception Handling[上篇]》
DECLARE 
@result 
INT

《谈谈基于SQL Server 的Exception Handling[上篇]》
SET 
@result 
= 
10
/
0

《谈谈基于SQL Server 的Exception Handling[上篇]》
PRINT 
@@ERROR

《谈谈基于SQL Server 的Exception Handling[上篇]》
Go

《谈谈基于SQL Server 的Exception Handling[上篇]》

会得到如下的执行结果:

《谈谈基于SQL Server 的Exception Handling[上篇]》

由于除零导致错误的产生,使@@ERROR的值变成8134。实际上8134是一个Error Number,代表某个系统定义的Error,每个预定义的Error由一个唯一的Error Number来唯一标识。@@ERROR本质上就是返回上一个语句执行遇到的ErrorID号,所以这并不是一个随机的值,无论是10/0还是100/0@@ERROR返回的结果都是一样的。

由于可以通过@@ERROR得到Error的类型,可以帮助我们在编写Stored Procedure的过程中,借助这个@@ERROR进行一些流程的控制。比如我们知道Error Number547代表违反外键约束,我们就可以通过@@ERROR的结果是否等于547来进行异常的处理:

 

 

《谈谈基于SQL Server 的Exception Handling[上篇]》UPDATE T_USERS_IN_ROLES SET [USER_ID] = dummy text
《谈谈基于SQL Server 的Exception Handling[上篇]》IF @@ERROR = 547
《谈谈基于SQL Server 的Exception Handling[上篇]》PRINT The user is not existent

《谈谈基于SQL Server 的Exception Handling[上篇]》

通过我们上面一个截图我们发现,在

SQL Server Management Studio
中执行任何一个语句的时候,如果遇到一个预定义的
Error
,会打印出相关的
Error Message
。这些
Error Message
到底是如何存储的呢?
Error message

Error Numbder
又是如何进行关联的呢?

 

实际上,SQL Server通过一个名为sys.messages的系统表来存储关于Error的一些信息[关于sys.messages,在后续的章节中还会提及]。下面是sys.messages的结构:message_id不仅仅代表message的唯一标识,对于一个预定义的Error,其Error number就是这个message_id。由于Localization的需要,我们需要为不同的语言定义不同的Message,这些Message共享一个message_id, 具体采用何种语言通过luange_id来标识。Severity代表Error的严重程度,我将在后续部分专门介绍。is_event_logged是一个Indicator,表明出现该Error是否需要在Event log中进行日志记录,text当然就是message文本了。

《谈谈基于SQL Server 的Exception Handling[上篇]》

我们可以通过下面的SQL来进行验证:

《谈谈基于SQL Server 的Exception Handling[上篇]》
DECLARE 
@result 
INT

《谈谈基于SQL Server 的Exception Handling[上篇]》
DECLARE 
@error 
INT

《谈谈基于SQL Server 的Exception Handling[上篇]》
SET 
@result 
= 
5
/
0

《谈谈基于SQL Server 的Exception Handling[上篇]》
SET 
@error 
= 
@@ERROR

《谈谈基于SQL Server 的Exception Handling[上篇]》
SELECT 
@error,sys.messages.
* 
FROM sys.messages 
WHERE message_id 
= 
@error

《谈谈基于SQL Server 的Exception Handling[上篇]》
Go

《谈谈基于SQL Server 的Exception Handling[上篇]》

下面是执行的结果:

《谈谈基于SQL Server 的Exception Handling[上篇]》

对于@@ERROR,有一点需要特别提醒的是:它仅仅代表前一个语句执行的Error Number,之后任何一段语句的执行都会改变@@ERROR的值,甚至是一个IF语句。我想通过下面一段SQL,你肯定会后一个深刻的认识:

《谈谈基于SQL Server 的Exception Handling[上篇]》

我们可以看到,一个简简单单的IF语句就将@@ERROR8134变成了0。不过想想也很简单,IF语句本身也是一个执行语句,在执行过程中并没有遇到Error,所以@@ERROR应该返回0。这也是我在上面的Sample中通过SET @error = @@ERROR@@ERROR进行预存的原因。

@@ERROR实际上代表的是在编写SQL或者Stored procedure中对异常的识别,大多数我们通过@@ERROR来判断一段SQL语句是否成功执行,保证没有遇到不可预知的异常。对于一些可以预知的异常在SQL中又该如何处理呢?

我发现很多程序员喜欢使用Output参数来处理这些预知的异常。比如:我们需要编写一个添加UserStored procedureuser name具有唯一性,添加一个和database中同名的user显然是不合法的,在很多情况下通过一个Output参数来返回操作最终执行的情况,比如:

《谈谈基于SQL Server 的Exception Handling[上篇]》CREATE Procedure P_USERS_I
《谈谈基于SQL Server 的Exception Handling[上篇]》    (
《谈谈基于SQL Server 的Exception Handling[上篇]》        @user_id            varchar(50),
《谈谈基于SQL Server 的Exception Handling[上篇]》        @user_name    nvarchar(256),
《谈谈基于SQL Server 的Exception Handling[上篇]》        @flag                INT OUTPUT
《谈谈基于SQL Server 的Exception Handling[上篇]》    )
《谈谈基于SQL Server 的Exception Handling[上篇]》AS
《谈谈基于SQL Server 的Exception Handling[上篇]》IF(EXISTS(SELECT * FROM  dbo.T_USERS WHERE LOWERED_USER_NAME = LOWER(@user_nameOR [USER_ID] = @user_id))
《谈谈基于SQL Server 的Exception Handling[上篇]》    BEGIN
《谈谈基于SQL Server 的Exception Handling[上篇]》        SET @flag = 1
《谈谈基于SQL Server 的Exception Handling[上篇]》        RETURN
《谈谈基于SQL Server 的Exception Handling[上篇]》    END     
《谈谈基于SQL Server 的Exception Handling[上篇]》INSERT INTO dbo.T_USERS
《谈谈基于SQL Server 的Exception Handling[上篇]》           ([USER_ID]
《谈谈基于SQL Server 的Exception Handling[上篇]》           ,[USER_NAME]
《谈谈基于SQL Server 的Exception Handling[上篇]》           ,LOWERED_USER_NAME)
《谈谈基于SQL Server 的Exception Handling[上篇]》     VALUES(@user_id@user_nameLOWER(@user_name))     
《谈谈基于SQL Server 的Exception Handling[上篇]》SET @flag = @@ERROR

《谈谈基于SQL Server 的Exception Handling[上篇]》

很显然通过flag output参数可以得到User的创建操作最终执行的结果:-1代表重名,0代表成功,大于0代表出现不可预知的异常。

说实话,我不太喜欢这样的异常处理方式,其实这并不是说这样的处理不好,大部分还是由于个人喜好决定。我觉得,既然添加一个同名的User本身代表一种Exception,从语义上讲,认为地抛出这样Exception的方式好像更加合理一点,所以我们喜欢通过调用RAISEERROR的方式将一个Error抛出。

二、       RAISEERROR

RAISEEROR是一个系统函数,用于奖某个可以预知的Exception抛出,供Application捕捉并处理,下面是RAISERROR的声明:

《谈谈基于SQL Server 的Exception Handling[上篇]》RAISERROR ( { msg_id | msg_str | @local_variable }
《谈谈基于SQL Server 的Exception Handling[上篇]》    { ,severity ,state }
《谈谈基于SQL Server 的Exception Handling[上篇]》    [ ,argument [ ,《谈谈基于SQL Server 的Exception Handling[上篇]》] ] )
《谈谈基于SQL Server 的Exception Handling[上篇]》[ WITH option [ ,《谈谈基于SQL Server 的Exception Handling[上篇]》] ]
《谈谈基于SQL Server 的Exception Handling[上篇]》


msg_id | msg_str | @local_variable
代表被你抛出的ErrorMessage,你可以同国3中方式来表示Messagemsg_id带面sys.messages中的message_id, msg_str表示一个自定义的文本,@local_variable则表示message的变量。

Severity一个代表严重程度的数字,其范围为0-25,其中0-18可以由任何用户指定,19-25只能由sysadmin指定。一般地,0-10为严重程度很低的错误,11-18来高级别的错误,19-25代表非常严重的错误,以致在执行完成之后会终止当前的Session

State一个0-127的整数,代表一个错误状态,对于在多个地方抛出Message一致的的情况,将State在不同的地方设置在不同的值,在Debug的时候可以很快知道是哪里出错了,所以State具有很现实的意义。

Argument向我们调用String.Format(string,…)一样,我们可以在一个一个参数中使用{0G}{1D}这样的站位符和进行格式处理的字符,这些站位符由后面的参数来填充。在这里也一样,在message中你一可以添加站位符,这着站位符由Argument来填充,具体如何定义,可以参阅SQL Server 2005 Books Online.

WITH option [ ,…n ]:代表一些额外的选项, LOG表示进行日志记录,NOWAIT表示立即将Message递交到客户端,SETERROR强制将当前真实的@@ERROR或者message_id返回到客户端。

明白了RAISERROR如何使用了后,我们可以修改我们的先前创建UserStored Procedure

《谈谈基于SQL Server 的Exception Handling[上篇]》CREATE Procedure P_USERS_I
《谈谈基于SQL Server 的Exception Handling[上篇]》    (
《谈谈基于SQL Server 的Exception Handling[上篇]》        @user_id            varchar(50),
《谈谈基于SQL Server 的Exception Handling[上篇]》        @user_name    nvarchar(256)
《谈谈基于SQL Server 的Exception Handling[上篇]》    )
《谈谈基于SQL Server 的Exception Handling[上篇]》AS
《谈谈基于SQL Server 的Exception Handling[上篇]》
《谈谈基于SQL Server 的Exception Handling[上篇]》IF(EXISTS(SELECT * FROM  dbo.T_USERS WHERE LOWERED_USER_NAME = LOWER(@user_nameOR [USER_ID] = @user_id))
《谈谈基于SQL Server 的Exception Handling[上篇]》    BEGIN
《谈谈基于SQL Server 的Exception Handling[上篇]》        RAISERROR (This role is already existent,16,1)
《谈谈基于SQL Server 的Exception Handling[上篇]》    END
《谈谈基于SQL Server 的Exception Handling[上篇]》
《谈谈基于SQL Server 的Exception Handling[上篇]》INSERT INTO dbo.T_USERS
《谈谈基于SQL Server 的Exception Handling[上篇]》           ([USER_ID]
《谈谈基于SQL Server 的Exception Handling[上篇]》           ,[USER_NAME]
《谈谈基于SQL Server 的Exception Handling[上篇]》           ,LOWERED_USER_NAME)
《谈谈基于SQL Server 的Exception Handling[上篇]》VALUES(@user_id@user_nameLOWER(@user_name))   

《谈谈基于SQL Server 的Exception Handling[上篇]》

[原创]谈谈基于SQL Server的Exception Handling – PART I

[原创]谈谈基于SQL Server 的Exception Handling – PART II

[原创]谈谈基于SQL Server 的Exception Handling – PART III 

作者:蒋金楠

微信公众账号:大内老A

微博:
www.weibo.com/artech

如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号(原来公众帐号
蒋金楠的自媒体将会停用)。

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
原文链接

点赞