zoukankan      html  css  js  c++  java
  • SQL Server 中关于 @@error 的一个小误区

    在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
  • 相关阅读:
    序列化和反序列化&持久化
    基于qiankun微前端的部署方案
    【MySQL】Explain执行计划 type类型说明
    【ElasticSearch】index read-only
    【MybatisPlus】Wrappers条件构造器构造or条件查询
    【布隆过滤器】基于Resisson的实现的布隆过滤器
    Nacos源码分析(三): 心跳设计
    Nacos源码分析(二):服务端和客户端实例注册
    Nacos源码分析(一): Nacos源码环境搭建
    【linux】如何在linux中查找文件、进程
  • 原文地址:https://www.cnblogs.com/seasons1987/p/3670107.html
Copyright © 2011-2022 走看看