设为首页 收藏本站
查看: 1269|回复: 0

[经验分享] SQL 2008 RAISERROR语法在SQL 2012/2014不兼容问题

[复制链接]

尚未签到

发表于 2017-7-14 20:49:32 | 显示全部楼层 |阅读模式
  原文 旧的RAISERROR语法在SQL 2012不兼容问题
  raiserror 写法:
  SQL 2008: raiserror 55030 'text error'
  SQL 2012: raiserror ('text error', 16, 1)





--最近有一支Store Procedure,里面有使用RAISERROR。而操作的方式是执行它后,并读取@@ERROR的值来判断有没有错误,如下,

CREATE PROC RaiseErrorTest
AS
BEGIN
RAISERROR 13001 'this is 13001 error';
END
GO

DECLARE @errNum INT;
SET @errNum = 0;

EXEC RaiseErrorTest;
SET @errNum = @@ERROR;
--如果没有错误,就commit交易
IF @errNum = 0
BEGIN
PRINT '执行OK Commit'
END
ELSE
BEGIN
PRINT '执行失败 Rollback'
END
GO
--执行结果会显示「执行失败 Rollback ,如下,
--Msg 13001, Level 16, State 1, Procedure RaiseErrorTest, Line 4
--this is 13001 error
--执行失败 Rollback

--但是在SQL 2012中一执行到该Store Procedure,就发生以下的错误,
--Msg 102, Level 15, State 1, Procedure RaiseErrorTest, Line 4
--接近 '13001' 之处的语法不正确。

--原因是因为我们使用了以下的RAISERROR语法过期了(从Sybase来的,可参考:raiserror)
--而在SQL 2008R2就说未来版本不Support以上的写法(SQL 2012还真的给我不Support)! 请参考:Deprecated Database Engine Features in SQL Server 2008 R2
--不过,那为何那支Store Procedure,可以被建立到SQL 2012的DB之中呢?
--因为那个DB是后SQL 2005备份好之后,再Restore到SQL 2012上,所以如果要重新将该Store Procedure执行到该DB的话,就会发生同样的问题。
--知道了问题所在就要加以调整它,如下,

CREATE PROC RaiseErrorTest2
AS
BEGIN
RAISERROR('this is 13001 error', 10, 1);
END
GO

--再执行以下的SQL,

DECLARE @errNum INT;
SET @errNum = 0;

EXEC RaiseErrorTest2;
SET @errNum = @@ERROR;
IF @errNum = 0
BEGIN
PRINT '执行OK Commit'
END
ELSE
BEGIN
PRINT '执行失败 Rollback'
END
--但执行结果却显示「执行OK Commit」,天呀! 我的@@ERROR的值变成0了!如下,
--this is 13001 error
--执行OK Commit

--表示如果使用RAISERROR('this is 13001 error', 10, 1)的方式,@@ERROR的值却是0。
--于是想要用THROW,但这样呼叫的程序也要一并修改,就要调查程序中有多少使用到这种方式,然后加以调整,但是THROW又不能用在SQL 2012之前的版本。
--…
--于是笔者想到一个很瞎的作法,但却能暂时满足SQL 2005 ~ SQL 2012的这种透过 RAISERROR 来判断@@ERROR不为0的做法。
--就是使用RAISERROR(msg_id , 10, 1),因为使用msg_id的话,需要透过 sp_addmessage 将讯息加入SQL之中,如果没有的话,就会产生错误。
--所以利用这错误,@@ERROR就会变成不为0,呼叫的程序就可以判断了,如下,

CREATE PROC RaiseErrorTest3
AS
BEGIN
RAISERROR(50001, 10, 1);
END
GO

--再執行以下的SQL,

DECLARE @errNum INT;
SET @errNum = 0;
EXEC RaiseErrorTest3;
SET @errNum = @@ERROR;
IF @errNum = 0
BEGIN
PRINT '執行OK Commit'
END
ELSE
BEGIN
PRINT '執行失敗 Rollback'
END
--执行结果会显示「执行失败 Rollback」(只是@@ERROR的值变成了18054,而不是我们要的50001,但针对我们旧有「只判断@@ERROR不为0」的做法算是可以兼容过去),如下,
--Msg 18054, Level 16, State 1, Procedure RaiseErrorTest3, Line 4
--Error 50001, severity 10, state 1 was raised, but no message with that error number was found in sys.messages. If error is larger than 50000, make sure the user-defined message is added using sp_addmessage.
--执行失败 Rollback

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-393957-1-1.html 上篇帖子: MyBatis3 用log4j在控制台输出 SQL 下篇帖子: 转 Java防止SQL注入
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表