使用中经常用到@@Error来判断上一个语句是否执行成功,对此小结一下,可能有些不准确,欢迎指出。

1.1  介绍

SQL SERVER 中@@表示系统全局变量

(1)   返回执行的上一个 Transact-SQL 语句的错误号,如果执行没有错误,则返回 0 。

(2)   如果错误是 sys.messages 目录视图中的错误之一,则 @@ERROR 将包含 sys.messages.message_id 列中表示该错误的值。  可以在 sys.messages 中查看与 @@ERROR 错误号相关的文本信息。

(3)   由于 @@ERROR 在每一条语句执行后被清除并且重置,因此应在语句验证后立即查看它,或将其保存到一个局部变量中以备以后查看。

1.2  范例及使用

(1)  分析执行SQL出现错误,后续脚本的执行情况

执行语句:

UPDATE tbOrder SET OrderNo = '201605010008' WHERE OrderNo = '201605010001'

--执行上一步出现外键约束错误,并且继续执行下一步(547为约束错误)

PRINT @@ERROR  --输出错误号

UPDATE tbOrder SET OrderAmount = 'ABC' WHERE OrderNo = '201605010001'

--在执行上一步时出现类型异常,并且不执行下面的步骤

PRINT @@ERROR  --未输出

说明:第1个SQL出错后,仍然执行了后面的代码,输出了@@Error错误号,第2个SQL出错后直接终止后续执行。

说明执行SQL出现错误后,有的错误会直接终止后续执行,有的出现错误后仍然可以继续执行后续脚本

(2)  查询成功后,系统变量@@Error变为0

执行语句:

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

PRINT 'Delete: ' + CAST(@@ERROR AS VARCHAR(20))   --输出

SELECT 1

PRINT 'Select: ' + + CAST(@@ERROR AS VARCHAR(20))  --输出 Select:0

说明:执行SQL 成功后,@@Error参数被设置为0

(3)  通过Try Catch捕获错误,输出@@Error

a)  还是上面的删除操作,产生外键约束问题

执行语句:

BEGIN TRY

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

PRINT 'DELETE ' + CAST(@@ERROR AS VARCHAR(20))   --没有输出

END TRY

BEGIN CATCH

PRINT 'CATCH ' + CAST(@@ERROR AS VARCHAR(20))

PRINT ERROR_MESSAGE()

PRINT ERROR_SEVERITY()

PRINT ERROR_STATE()

END CATCH

GO

输出结果:

b)  还是上面的更新操作,产生错误,不执行后续代码

执行语句:

BEGIN TRY

UPDATE tbOrder SET OrderAmount = 'ABC' WHERE OrderNo = '201605010001'

PRINT 'UPDATE ' + CAST(@@ERROR AS VARCHAR(20))

END TRY

BEGIN CATCH

PRINT 'CATCH ' + CAST(@@ERROR AS VARCHAR(20))

PRINT ERROR_MESSAGE()

PRINT ERROR_SEVERITY()

PRINT ERROR_STATE()

END CATCH

GO

输出结果:

说明:在Try中执行SQL产生错误后,直接被Catch捕获,Try中后续代码终止执行,直接执行Catch中的代码

(4)  测试有事物的操作是否产生错误后都回滚

a)  直接增加事物

执行脚本:

BEGIN TRAN

--执行正常

INSERT INTO dbo.tbOrderDetail( OrderNo, ProductID, Quantity, Price )

VALUES  ( '201605010001',50,200,10)

--执行报错

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

PRINT 'DELETE: ' +  CAST(@@ERROR AS VARCHAR(20))

COMMIT TRAN

执行结果(同时查询订单明细):

说明:执行出错后,前面执行的插入操作数据并没有回滚,说明直接增加事物并不能回滚出错前的数据

b)  增加@@Error判断执行是否成功

执行脚本:

BEGIN TRAN

--执行正常

INSERT INTO dbo.tbOrderDetail( OrderNo, ProductID, Quantity, Price )

VALUES  ( '201605010001',50,200,10)

IF @@ERROR <> 0

BEGIN

ROLLBACK TRAN

RETURN

END

--执行报错

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

--UPDATE tbOrder SET OrderAmount = 'ABC' WHERE OrderNo = '201605010001'   --将Delete操作换成更新操作产生异常后事物也会回滚

IF @@ERROR <> 0

BEGIN

ROLLBACK TRAN

RETURN

END

COMMIT TRAN

执行结果(同时查询订单明细):

说明:增加@@Error判断,执行错误时前面的数据都回滚了。同时将语句中的Delete操作换成执行已屏蔽的更新操作产生异常后,事物同样也会回滚

(5)  通过设置XACT_ABORT,测试出现错误整个事物是否回滚

备注:当 SET XACT_ABORT 为 ON 时,如果执行 Transact-SQL 语句产生运行时错误,则整个事务将终止并回滚。

当 SET XACT_ABORT 为 OFF 时,有时只回滚产生错误的 Transact-SQL 语句,而事务将继续进行处理。如果错误很严重,那么即使 SET XACT_ABORT 为 OFF,也可能回滚整个事务。OFF 是默认设置。

a)  XACT_ABORT 为 OFF,只有执行出错的回滚

执行脚本:

SET XACT_ABORT OFF

BEGIN TRAN

--执行正常

INSERT INTO dbo.tbOrderDetail( OrderNo, ProductID, Quantity, Price )

VALUES  ( '201605010001',50,200,10)

--执行报错

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

PRINT 'DELETE: ' +  CAST(@@ERROR AS VARCHAR(20))

--执行正常

INSERT INTO dbo.tbOrderDetail( OrderNo, ProductID, Quantity, Price )

VALUES  ( '201605010001',51,10,100)

COMMIT TRAN

执行结果(同时查询明细):

b)  XACT_ABORT 为 ON,强制整个事物回滚

执行脚本:

SET XACT_ABORT ON

BEGIN TRAN

--执行正常

INSERT INTO dbo.tbOrderDetail( OrderNo, ProductID, Quantity, Price )

VALUES  ( '201605010001',50,200,10)

--执行报错

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

PRINT 'DELETE: ' +  CAST(@@ERROR AS VARCHAR(20))

COMMIT TRAN

SET XACT_ABORT OFF

执行结果(同时查询明细):

说明:设置XACT_ABORT为OFF(默认为OFF),只有执行出错的语句回滚了,其他的没有回滚数据;设置XACT_ABORT为ON时,当执行SQL产生错误后,强制整个事物回滚,并且不在执行后续代码

(6)  通过Try Catch捕获错误,显示事物执行错误,并回滚

执行脚本:

BEGIN TRAN

BEGIN TRY

--执行正常

INSERT INTO dbo.tbOrderDetail( OrderNo, ProductID, Quantity, Price )

VALUES  ( '201605010001',50,200,10)

--执行报错

DELETE FROM tbOrder WHERE OrderNo = '201605010001'

--UPDATE tbOrder SET OrderAmount = 'ABC' WHERE OrderNo = '201605010001'

COMMIT TRAN

END  TRY

BEGIN CATCH

PRINT '执行错误' + CAST(@@ERROR AS VARCHAR(20))

PRINT ERROR_MESSAGE()

PRINT ERROR_SEVERITY()

PRINT ERROR_STATE()

ROLLBACK TRAN --必须增加回滚,否则事物会一直挂死在哪里

END CATCH

执行结果(同时查询订单明细):

1.3  总结说明

参照上面的使用演示及结果,可以由如下3种处理方式确保数据完整性

(1) 通过@@Error判断来语句是否执行成功,是否事物需要回滚(参见1.2 (4) )

(2) 设置XACT_ABORT为ON,强制整个事物执行出错时都回滚 (参见1.2 (5) )

(3) 通过Try Catch捕获执行异常,并回滚事物 (参见1.2 (6) )

1.4  参考资料

微软官网解释

https://msdn.microsoft.com/zh-cn/library/ms188790.aspx

try catch 捕获不到的一些错误及解决方法

http://blog.csdn.net/kk185800961/article/details/40043415

事务执行情况跟踪分析

http://blog.csdn.net/zhaowenzhong/article/details/16342843

1.5  相关附件

相关SQL脚本

@@Error使用简单小结的更多相关文章

  1. 前台vue的使用简单小结

    前台vue的使用简单小结 本项目要求:安装有node.js 6.0以及以上安装npm使用vue.js官方安装方法初始化项目npm install安装VueResurce:npm install vue ...

  2. springboot微服务的简单小结

    springboot微服务的简单小结 近来公司用springboot微服务,所以小结一下. 基础: 什么是SpingBoot微服务? 如何创建SpringBoot微服务? 如何管理和完善SpringB ...

  3. ASP.NET MVC ViewData/ViewBag 简单小结

    近期在项目中遇到一个问题,就是用ViewBag.Model存储匿名对象传递给View,但是需要根据条件给匿名对象添加属性,这个可真心不易,Google了一下发现很多方案都是动态编译神马的,感觉好高大上 ...

  4. git知识简单小结

    git特点: 1)分布式 2)存储快照而非差异 3)本地有完全的版本库,几乎所有操作都在本地 4)有内在的一致性,SHA1 5)优秀的分支管理 6)支持各种协同模式 7)开源,有一些第三方软件可整合使 ...

  5. SpringMVC_简单小结

    SpringMVC是一个简单的.优秀的框架.应了那句话简单就是美,而且他强大不失灵活,性能也很优秀. 机制:spring mvc的入口是servlet,而struts2是filter(这里要指出,fi ...

  6. log4c 编译安装简单小结(ubuntu12)

    1 下载源码,解压 (假定解压到了当前用户的根目录下,位置是~/log4c-1.2.3) 2 编译安装log4c(指定--prefix极其重要,如果没有指定到/usr下会有一堆麻烦事,还有,不能按照他 ...

  7. 微信小程序支付简单小结与梳理

    前言 公司最近在做微信小程序,被分配到做支付这一块,现在对这一块做一个简单的总结和梳理. 支付,对于购物来说,可以说是占据了十分重要的一块,毕竟能收到钱才是重点. 当然在开发之前,我们需要有下面这些东 ...

  8. 关于Python中的ifelse、while和for循环的简单小结

    1.ifelse 1.1首先简单编辑一个关于ifelse的程序: _username = 'yanfeixu' _password = 'wuyifan' username = input(" ...

  9. java post请求的表单提交和json提交简单小结

    在java实现http请求时有分为多种参数的传递方式,以下给出通过form表单提交和json提交的参数传递方式: public String POST_FORM(String url, Map< ...

随机推荐

  1. MyEclipse 8.5 开发环境配置,汉化,Aptana2.0插件,SVN 插件,Flex Builder 3/4 插件安装(转载)

    转载地址http://elf8848.iteye.com/blog/630864 下载MyEclipse 8.5 可以通过代理http://www.proxyie.cn/访问MyEclipse的官方网 ...

  2. 如何查询redhat的版本信息

    cat /etc/redhat-release lsb_release -a

  3. 内网能PING通TELNET通不能访问解决

    遇到一个离奇故障,内网,两个主机在同一IP段内,能互相PING通,TELNET对方的WEB服务器端口,通. 但用IE访问时不能,显示HTTP400.这明显是客户端系统的问题啊!但如何解决呢?我强烈怀疑 ...

  4. 58 web框架Argo代码分析

    贴地址:https://github.com/58code/Argo 核心jar javax.servlet-api 3.0.1 guice 3.0 velocity 1.7 框架使用 servlet ...

  5. C# 实现对网站数据的采集和抓取

    首先大家需要清楚一点的是:任何网站的页面,无论是php.jsp.aspx这些动态页面还是用后台程序生成的静态页面都是可以在浏览器中查看其HTML源文件的. 所以当你要开发数据采集程序的时候,你必须先对 ...

  6. FindMe

    https://github.com/hongdong/FindMe_Android https://github.com/hongdong/FindMe_Server https://github. ...

  7. JavaScript宝座:七大框架论剑

    JavaScript宝座:七大框架论剑 一周前,Throne of JS大会在多伦多召开,这应该是我参加过的最有料也最不一样的一次大会.大会官网如是说: 加载整个页面,然后再“渐进增强”以添加动态行为 ...

  8. js正則表達式语法

    1. 正則表達式规则 1.1 普通字符 字母.数字.汉字.下划线.以及后边章节中没有特殊定义的标点符号,都是"普通字符".表达式中的普通字符,在匹配一个字符串的时候,匹配与之同样的 ...

  9. Android中的Handler的具体用法

    Android UI操作并不是线程安全的并且这些操作必须在UI线程中执行.Android利用Handler来实现UI线程的更新的. Handler是Android中的消息发送器,其在哪个Activit ...

  10. NSNotificationCenter通知中心

    概述 NSNotificationCenter通知中心,通常用于一对一或者一对多的消息传递,即当一个地方改变时,要求改变其他的一些地方,例如当网络请求回来了新的数据,需要刷新本地信息和本地内存里面的界 ...