SQL Server笔记 -- 第20章:TRY/CATCH

第20.1节:TRY/CATCH中的事务

这将由于无效的datetime而回滚两个插入

sql 复制代码
BEGIN TRANSACTION
BEGIN TRY
    INSERT INTO dbo.Sale(Price, SaleDate, Quantity)
    VALUES (5.2, GETDATE(), 1)
    INSERT INTO dbo.Sale(Price, SaleDate, Quantity)
    VALUES (5.2, 'not a date', 1)
    COMMIT TRANSACTION
END TRY
BEGIN CATCH
    ROLLBACK TRANSACTION --首先回滚,然后抛出
    THROW
END CATCH

这将提交两个插入:

sql 复制代码
BEGIN TRANSACTION
BEGIN TRY
    INSERT INTO dbo.Sale(Price, SaleDate, Quantity)
    VALUES (5.2, GETDATE(), 1)
    INSERT INTO dbo.Sale(Price, SaleDate, Quantity)
    VALUES (5.2, GETDATE(), 1)
    COMMIT TRANSACTION
END TRY
BEGIN CATCH
    THROW
    ROLLBACK TRANSACTION
END CATCH

第20.2节:在try-catch块中引发错误

RAISERROR函数将在TRY CATCH块中生成错误:

sql 复制代码
DECLARE @msg nvarchar(50) = 'Here is a problem!'
BEGIN TRY
    print 'First statement';
    RAISERROR(@msg, 11, 1);
    print 'Second statement';
END TRY
BEGIN CATCH
    print 'Error:' + ERROR_MESSAGE();
END CATCH

第二个参数大于10(本例中为11)的RAISERROR将停止TRY BLOCK中的执行并引发将在CATCH块中处理的错误。您可以使用ERROR_MESSAGE()函数访问错误消息。

此示例的输出为:

复制代码
First statement
Error:Here is a problem!

第20.3节:在try catch块中引发信息消息

第二个参数(严重级别)小于或等于10的RAISERROR不会引发异常。

sql 复制代码
BEGIN TRY
    print 'First statement';
    RAISERROR('Here is a problem!', 10, 15);
    print 'Second statement';
END TRY
BEGIN CATCH
    print 'Error:' + ERROR_MESSAGE();
END CATCH

在RAISERROR语句之后,第三条语句将被执行,CATCH块不会被调用。执行结果为:

复制代码
First statement
Here is a problem!
Second statement

第20.4节:重新引发RAISERROR生成的异常

您可以使用THROW语句重新引发在CATCH块中捕获的错误:

sql 复制代码
DECLARE @msg nvarchar(50) = 'Here is a problem! Area:''%s'' Line:''%i'''
BEGIN TRY
    print 'First statement';
    RAISERROR(@msg, 11, 1, 'TRY BLOCK', 2);
    print 'Second statement';
END TRY
BEGIN CATCH
    print 'Error:' + ERROR_MESSAGE();
    THROW;
END CATCH

请注意,在这种情况下,我们使用格式化参数(第四个和第五个参数)引发错误。如果您想在消息中添加更多信息,这可能很有用。

执行结果为:

复制代码
First statement
Error:Here is a problem! Area:'TRY BLOCK' Line:'2'
Msg 50000, Level 11, State 1, Line 26
Here is a problem! Area:'TRY BLOCK' Line:'2'

第20.5节:在TRY/CATCH块中引发异常

您可以在try catch块中引发异常:

sql 复制代码
DECLARE @msg nvarchar(50) = 'Here is a problem!'
BEGIN TRY
    print 'First statement';
    THROW 51000, @msg, 15;
    print 'Second statement';
END TRY
BEGIN CATCH
    print 'Error:' + ERROR_MESSAGE();
    THROW;
END CATCH

异常将在CATCH块中处理,然后使用不带参数的THROW重新引发。

复制代码
First statement
Error:Here is a problem!
Msg 51000, Level 16, State 15, Line 39
Here is a problem!

THROW与RAISERROR的相似之处和区别:

  • 建议新应用程序使用THROW而不是RAISERROR。
  • THROW可以使用任何数字作为第一个参数(错误号),RAISERROR只能使用sys.messages视图中的id
  • THROW的严重级别为16(无法更改)
  • THROW不能像RAISERROR那样格式化参数。如果需要,请使用FORMATMESSAGE函数作为RAISERROR的参数
相关推荐
二哈赛车手1 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
栗子~~2 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8292 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
星星也在雾里3 小时前
PgBouncer 解决 PostgreSQL 连接数超限 + 可视化监控
数据库·postgresql
Maimai108083 小时前
React如何用 @microsoft/fetch-event-source 落地 SSE:比原生 EventSource 更灵活的实时推送方案
前端·javascript·react.js·microsoft·前端框架·reactjs·webassembly
未若君雅裁3 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
AI人工智能+电脑小能手4 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
xian_wwq4 小时前
【学习笔记】AGC协调控制系统概述
笔记·学习
阿维的博客日记4 小时前
Nacos 为什么能让配置动态生效?(涉及 @RefreshScope 注解)
java·spring
雨辰AI4 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务