首页 > SQL Server 中关于 @@error 的一个小误区

SQL Server 中关于 @@error 的一个小误区

原文:SQL Server 中关于 @@error 的一个小误区

在SQL Server中,我常常会看到有些前辈这样写:

if(@@error<>0)ROLLBACK TRANSACTION T
elseCOMMIT TRANSACTION T

一开始,我看见别人这么写,我就想当然的以为它只是个计数器,每当检测到一处错误时,@@error的值+1,不过就因为这个理所当然,所以杯具了...

实际上,它并不是一个计数器,它是一个动态的值,动态的标识最后一条SQL命令执行的结果,如果成功则为0,不成功则标识错误码。所以,像上面这种写法是不妥的,举个例子,如下:

SET NOCOUNT ON;
SET XACT_ABORT ON;  --执行 Transact-SQL 语句产生运行时错误,则整个事务将终止并回滚BEGIN TRANSACTION TUPDATE Test
SET a='已更新'
WHERE a='未更新'RAISERROR ('不好意思,你没有权限!',16,1)



SELECT GETDATE()if(@@error<>0)ROLLBACK TRANSACTION T elseCOMMIT TRANSACTION T

分析:

  按我以前的理解来说,【 RAISERROR ('不好意思,你没有权限!',16,1) 】这里抛出了一个错误,整个事务应该回滚才对,可是,它却没有回滚!!那么原因出在哪呢?原来,问题出在"SELECT GETDATE()"这句上面!因为执行RAISERROR语句时,@@error的值不为0(好像是5000),而当执行到下一句"SELECT GETDATE()"时,@@error的值又变为0了!所以,后面的if语句自然没有捕捉到任何错误...

 

对策:

  既然找到了原因,那解决办法自然也少不了。用Try...CATCH语法就可以了,语句如下:

SET NOCOUNT ON;
SET XACT_ABORT ON;  --执行 Transact-SQL 语句产生运行时错误,则整个事务将终止并回滚BEGIN TRYBEGIN TRANSACTION TUPDATE TestSET a='已更新'WHERE a='未更新'RAISERROR ('不好意思,你没有权限!',16,1)SELECT GETDATE()COMMIT TRANSACTION T
END TRY
BEGIN CATCHDECLARE @msg nvarchar(2000)=ERROR_MESSAGE()    --将捕捉到的错误信息存在变量@msg中               RAISERROR (@msg,16,1)    --此处才能抛出(好像是这样子....)ROLLBACK TRANSACTION T  --出錯回滾事務
END CATCH

 

 

更多相关:

  • 1. set的初始化 set可以这样初始化 set iset{ 1,2,3 }; set iset2 = { 1,2,3 }; 在初始化set的时候,若出现重复的元素 set iset3{ 1,2,3,3,3 }; set iset4 = { 1,2,3,3,3 };...

  •     Set 对象存储的值总是唯一的 Set 对象方法 方法描述add添加某个值,返回Set对象本身。clear删除所有的键/值对,没有返回值。delete删除某个键,返回true。如果删除失败,返回false。forEach对每个元素执行指定操作。has返回一个布尔值,表示某个键是否在当前 Set 对象之中。 Set 对象...

  • 我现在的vimrc配置文件 runtime! debian.vim "设置编码 set encoding=utf-8 set fencs=utf-8,ucs-bom,shift-jis,gb18030,gbk,gb2312,cp936 set fileencodings=utf-8,ucs-bom,chinese"语言设置...

  • ACCEPT acm作业 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=204 因为老师是在集合那里要我们做这道题。所以我很是天真的就以为要用集合做,结果发现网上都是用数组简单明了地实现了,显得我的代码,特么地超级恶心!!!!!!! 在这里存档一下,别人就不要看了...

  • USE [svnhost]GO/****** 对象:  StoredProcedure [dbo].[up_Page2005]    脚本日期: 05/21/2008 11:27:05 ******/SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGOCREATE proc [dbo].[up_P...