在SQL Server中,我常常会看到有些前辈这样写:
if
(
@@error
<>
0
)
ROLLBACK
TRANSACTION
T
else
COMMIT
TRANSACTION
T
一开始,我看见别人这么写,我就想当然的以为它只是个计数器,每当检测到一处错误时,@@error的值+1,不过就因为这个理所当然,所以杯具了...
实际上,它并不是一个计数器,它是一个动态的值,动态的标识最后一条SQL命令执行的结果,如果成功则为0,不成功则标识错误码。所以,像上面这种写法是不妥的,举个例子,如下:
SET
NOCOUNT
ON
;
SET
XACT_ABORT
ON
;
--
执行 Transact-SQL 语句产生运行时错误,则整个事务将终止并回滚
BEGIN
TRANSACTION
T
UPDATE
Test
SET
a
=
'
已更新
'
WHERE
a
=
'
未更新
'
RAISERROR
(
'
不好意思,你没有权限!
'
,
16
,
1
)
SELECT
GETDATE
()
if
(
@@error
<>
0
)
ROLLBACK
TRANSACTION
T
else
COMMIT
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
TRY
BEGIN
TRANSACTION
T
UPDATE
Test
SET
a
=
'
已更新
'
WHERE
a
=
'
未更新
'
RAISERROR
(
'
不好意思,你没有权限!
'
,
16
,
1
)
SELECT
GETDATE
()
COMMIT
TRANSACTION
T
END
TRY
BEGIN
CATCH
DECLARE
@msg
nvarchar
(
2000
)
=
ERROR_MESSAGE()
--
将捕捉到的错误信息存在变量@msg中
RAISERROR
(
@msg
,
16
,
1
)
--
此处才能抛出(好像是这样子....)
ROLLBACK
TRANSACTION
T
--
出錯回滾事務
END
CATCH

