SQL Server异常代码处理的深入讲解
前言
SQLServer使用TRY...CATCH结构实现TSQL语句的错误处理,TRY命令负责监控语句执行的情况,如果有TSQL语句发生异常,并且严重级别(SeverityLevel)大于10,并且小于20,那么CATCH命令会捕获到异常的错误。
BEGINTRY {sql_statement|statement_block} ENDTRY BEGINCATCH [{sql_statement|statement_block}] ENDCATCH
数据库开发工程师需要把一条或多条语句写入到TRY代码块中,如果TRY代码块中的代码在执行过程中发生错误,那么在错误发生的点之后的代码不再执行,程序的控制权转移到CATCH代码块块中。如果TRY代码块没有发生错误,那么不会执行CATCH代码块,程序的控制权转移到ENDCATCH之后的语句。
注意,在CATCH代码块中捕获的异常不会返回给调用程序,如果需要把错误消息返回给调用程序,需要在CATCH代码块中使用THROW(或RAISERROR,不推荐使用)命令显式抛出错误。
一,获取异常消息
在TSQL中,使用TRY和CATCH编写异常处理代码块,在CATCH子句中,使用以下函数,能够获取异常发生时的信息。
--返回发生错误的代码行号(LineNumber) ERROR_LINE() --返回错误号(ErrorNumber) ERROR_NUMBER() @@ERROR --返回错误消息(ErrorMessage) ERROR_MESSAGE() --返回发生错误的SPName ERROR_PROCEDURE() --返回错误的严重度(ErrorSeverity) ERROR_SEVERITY() --返回错误的状态(ErrorState) ERROR_STATE()
SQLServer抛出的一个错误,通常包括错误代码(ErrorNumber)、严重级别(SeverityLevel)、错误状态(ErrorState)和错误消息(ErrorMessage)等信息。
1,错误代码
错误代码,可以由变量@@ERROR和函数ERROR_NUMBER()获得,用于返回上一条语句的错误代码,该代码唯一标识该错误。
2,错误的严重级别
错误的严重程序(SeverityLevel)共有24个级别,表明SQLSever遇到问题的类型,SeverityLevel是一个int类型,可以由函数ERROR_SEVERITY()返回,数值越大,说明问题越严重。
按照错误对系统的影响程序,把严重级别分为四组:
- 0-10:信息,可以认为是warning
- 11-16:错误,是用户代码导致的
- 17-19:非常严重的错误,只能由系统管理员来修复
- 20-24:致命的错误,可能导致整个系统无法正常使用
从17-19,错误不能被用户修正,只能由系统管理员来修复问题。
从20-24,这个级别的错误遇到的情况比较少,一旦遇到,那么基本上表明整个数据库系统遇到了非常严重的错误:
3,错误状态
错误状态(ErrorState)是用户自定义的编码,用于使开发者能够轻易识别引起异常的确切位置。
4,错误消息
错误消息,是关于错误的描述性文本,可以是SQLServer系统预定义的错误信息,也可以是THROW命令抛出的用户自定义的文本。
二,抛出异常消息
在SQLServer2012及之后的版本中,使用Throw关键字代替RAISERROR,用于抛出异常,并将执行控制权转移到Catch代码块。
THROW[error_number,error_message,error_state];
参数注释:
- error_number:错误代码,是一个int类型,数值必须大于5000,小于2147483647,这是用户自定义的错误代码。
- error_message:错误消息,类型是nvarchar(2048)
- state:跟错误相关联的一个state,类型是tinyint,取值范围是:0-255
注意:在THROW语句之前的语句,必须以分号;结尾。
当THROW语句用于抛出自定义的异常时,severtylevel常常被设置为默认的16;当THROW用于re-throw,此时THROW没有任何参数,处于CATCH代码块中,仅仅用于把CATCH捕获的异常重新抛出,severtylevel,state,错误消息跟原始异常相同。
例子1,抛出自定义的异常:
BEGINTRY SELECT1/0 ENDTRY BEGINCATCH ;THROW51000,'Dividebyzeroerrorencountered',1; ENDCATCH;
SQLServer抛出的异常消息是,自定义的错误代码是51000,严重级别(SeverityLevel)是16,错误状态是1,错误行是5:
Msg51000,Level16,State1,Line5
Dividebyzeroerrorencountered
例子2,重抛异常,把系统检测到的错误从Catch代码块中抛出:
BEGINTRY SELECT1/0 ENDTRY BEGINCATCH ;THROW; ENDCATCH;
SQLServer抛出的异常消息是,错误代码是8134,严重级别(SeverityLevel)是16,错误状态是1,错误行是2:
Msg8134,Level16,State1,Line2
Dividebyzeroerrorencountered.
三,不受TRY...CATCH结构影响的错误
TRY...CATCH只捕获严重级别从11到19的错误,不会捕获严重级别是1-10,20-24的错误。
如果session会系统管理员使用KILL命令杀掉,那么TRY...CATCH结构不会捕获。
四,在事务中处理异常
如果在TRY代码块生成的错误,导致当前事务的状态变成无效,那么该事务就是不可提交的事务(uncommittabletransaction)。一个不可提交的事务,只能执行read操作,或者回滚(ROLLBACKTRANSACTION),不能执行TSQL语句来执行写操作,该事务也不能提交。函数XACT_STATE()返回-1表示,当前的事务是不可提交的事务;返回1表示当前的事务是可以提交的。数据库开发人员需要通过XACT_STATE()来对事务执行提交或回滚的操作。
例如,在事务中处理异常,可以参考以下代码,在实际应用程序,可以把异常信息记录在数据表中,便于进行故障排除:
--SETXACT_ABORTONwillrenderthetransactionuncommittablewhentheconstraintviolationoccurs. SETXACT_ABORTON; BEGINTRY BEGINTRANSACTION; --AFOREIGNKEYconstraintexistsonthistable.Thisstatementwillgenerateaconstraintviolationerror. DELETEFROMProduction.Product WHEREProductID=980; --Ifthedeleteoperationsucceeds,committhetransaction.TheCATCHblockwillnotexecute. COMMITTRANSACTION; ENDTRY BEGINCATCH --TestXACT_STATEfor0,1,or-1. --If1,thetransactioniscommittable. --If-1,thetransactionisuncommittableandshouldberolledback. --XACT_STATE=0meansthereisnotransactionandacommitorrollbackoperationwouldgenerateanerror. --Testwhetherthetransactionisuncommittable. IF(XACT_STATE())=-1 BEGIN --LoggingExceptioninfo,asthetransactionisinanuncommittablestate.Rollingbacktransaction. SELECT ERROR_NUMBER()ASErrorNumber, ERROR_SEVERITY()ASErrorSeverity, ERROR_STATE()ASErrorState, ERROR_PROCEDURE()ASErrorProcedure, ERROR_LINE()ASErrorLine, ERROR_MESSAGE()ASErrorMessage; ROLLBACKTRANSACTION; END; --Testwhetherthetransactionisactiveandvalid. IF(XACT_STATE())=1 BEGIN --'Thetransactioniscommittable.Committingtransaction.' COMMITTRANSACTION; END; ENDCATCH;
总结
到此这篇关于SQLServer异常代码处理的文章就介绍到这了,更多相关SQLServer异常代码处理内容请搜索毛票票以前的文章或继续浏览下面的相关文章希望大家以后多多支持毛票票!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。