SQL Server 异常代码处理

TSQL使用TRY...CATCH代码块来实现错误处理,把一条或多条语句写入到TRY代码块中,由TRY来负责监控,由CATCH捕获错误,并对错误进行处理。

BEGIN TRY  
     { sql_statement | statement_block }  
END TRY  
BEGIN CATCH  
     [ { sql_statement | statement_block } ]  
END CATCH  

 如果TRY代码块中的代码在执行过程中发生错误,那么在错误发生的点之后的代码不再执行,程序的控制权转移到CATCH代码块块中。如果TRY代码块没有发生错误,那么不会执行CATCH代码块,程序的控制权转移到END CATCH之后的语句。

一,获取异常消息

在TSQL中,使用try 和 catch编写异常处理代码块,在catch子句中,使用以下函数,能够获取异常发生时的信息。

--返回发生错误的代码行号(LineNumber)
ERROR_LINE ( ) 
--返回错误号(ErrorNumber)
ERROR_NUMBER ( ) 
@@ERROR 
--返回错误消息(ErrorMessage)
ERROR_MESSAGE ( ) 
--返回发生错误的SP Name
ERROR_PROCEDURE ( ) 
--返回错误的严重度(Error Severity)
ERROR_SEVERITY ( ) 
--返回错误的状态(Error State)
ERROR_STATE()

一个错误,通常包括错误代码(Error Number)、严重级别(Severity Level)、错误状态(Error State)和错误消息(Error Message)等信息。

1,错误代码

错误代码,可以由@ERROR 变量和ERROR_NUMBER()函数获得,用于返回上一条语句的错误代码。

2,错误的严重级别

错误的严重程序(Severity Level)共有24个级别,表明SQL Sever遇到的问题的类型,Severity Level是一个int类型,可以由函数ERROR_SEVERITY() 返回。

SQL Server 异常代码处理_第1张图片

从17-19,错误不能被用户修正,只能由系统管理员来修复问题。

SQL Server 异常代码处理_第2张图片

从20-24,这个级别的错误遇到的情况比较少,一旦遇到,那么基本上表明整个数据库系统遇到了非常严重的错误:

SQL Server 异常代码处理_第3张图片

3,错误状态

错误状态(Error State)是用户自定义的编码,用于使开发者能够轻易识别引起异常的确切位置。

4,错误消息

错误消息,是关于错误的描述性的文本,可以是SQL Server系统预定义的错误信息,也可以通过THROW命令抛出用户自定义的文本。

5,例子,在事务中实现异常处理

在进行调试时,可以参考以下示例代码,把异常信息记录在数据表中,以便进行代码的故障排除

-- SET XACT_ABORT ON will render the transaction uncommittable when the constraint violation occurs.
SET XACT_ABORT ON;  
  
BEGIN TRY  
    BEGIN TRANSACTION;  
        -- A FOREIGN KEY constraint exists on this table. This statement will generate a constraint violation error.
        DELETE FROM Production.Product  
        WHERE ProductID = 980;  
    -- If the delete operation succeeds, commit the transaction. The CATCH block will not execute.
    COMMIT TRANSACTION;  
END TRY  
BEGIN CATCH  
    -- Test XACT_STATE for 0, 1, or -1.  
    -- If 1, the transaction is committable.  
    -- If -1, the transaction is uncommittable and should be rolled back.
    -- XACT_STATE = 0 means there is no transaction and a commit or rollback operation would generate an error.
  
    -- Test whether the transaction is uncommittable.
    IF (XACT_STATE()) = -1  
    BEGIN  
        --Logging Exception info, as the transaction is in an uncommittable state. Rolling back transaction.
        SELECT  
            ERROR_NUMBER() AS ErrorNumber,  
            ERROR_SEVERITY() AS ErrorSeverity,  
            ERROR_STATE() AS ErrorState,  
            ERROR_PROCEDURE() AS ErrorProcedure,  
            ERROR_LINE() AS ErrorLine,  
            ERROR_MESSAGE() AS ErrorMessage;   
        ROLLBACK TRANSACTION;  
    END;  
    -- Test whether the transaction is active and valid.  
    IF (XACT_STATE()) = 1  
    BEGIN  
        --'The transaction is committable. Committing transaction.'  
        COMMIT TRANSACTION;     
    END;  
END CATCH; 
View Code

二,抛出异常消息

在SQL Server 2012及之后的版本中,使用 Throw 关键字代替RaiseError,用于抛出异常,并将执行控制权转移到Catch 代码块。

THROW [error_number, error_message, error_state];

参数注释:

  • error_number:错误代码,是一个int类型,数值必须大于5000,小于 2147483647,这是用户自定义的错误代码。
  • error_message:错误消息,类型是nvarchar(2048)
  • state:跟错误相关联的一个state,类型是tinyint,取值范围是:0-255

注意:在THROW语句之前的语句,必须以分号; 结尾。

当THROW语句用于抛出自定义的异常时,severty level 常常被设置为默认的16;当THROW用于re-throw,此时THROW 没有任何参数,处于CATCH代码块中,仅仅用于把CATCH捕获的异常重新抛出,severty level,state,错误消息跟原始异常相同。

例子1,抛出自定义的异常:

BEGIN TRY  
    SELECT 1/0
END TRY  
BEGIN CATCH  
    ;THROW 51000, 'Divide by zero error encountered', 1;   
END CATCH; 

SQL Server抛出的异常消息是,自定义的错误代码是51000,严重级别(Severity Level)是16,错误状态是1,错误行是5:

Msg 51000, Level 16, State 1, Line 5
Divide by zero error encountered

例子2,重抛异常,把系统检测到的错误从Catch代码块中抛出:

BEGIN TRY  
    SELECT 1/0
END TRY  
BEGIN CATCH  
    ;THROW;   
END CATCH; 

SQL Server抛出的异常消息是,错误代码是8134,严重级别(Severity Level)是16,错误状态是1,错误行是2:

Msg 8134, Level 16, State 1, Line 2
Divide by zero error encountered.

 

 

参考文档:

 

你可能感兴趣的:(SQL Server 异常代码处理)