gpt4 book ai didi

sql-server-2008 - T-SQL 是否在 "BEGIN CATCH" block 中完全关闭了错误处理?

转载 作者:行者123 更新时间:2023-12-03 07:40:54 24 4
gpt4 key购买 nike

我只想确认我所看到的。下面是我写的概念存储过程的一个小证明,它说明了我在一个更大的问题中遇到的问题。

ALTER PROCEDURE TestErrorHandling
@Param1 varchar(1) = ''
AS
BEGIN
/*
Unit test:
DECLARE @returnStatus nvarchar(15);
Exec @returnStatus = TestErrorHandling
print @returnStatus
*/

BEGIN TRY
print 'Start'
IF @Param1 = '' raiserror( '@Param1 is missing', 18, 1 );
print 'Should not see this'

END TRY
BEGIN CATCH
print error_message()
print error_state()
print error_number()
INSERT INTO [FlightOrderUploadFailedLog]
(EvtTyp)
VALUES ('max size of EvtType is only varchar(15) so this should cause truncation')
print 'after insert '
print error_message()
print error_state()
print error_number()
return -- added in second version (after original post)

END CATCH

END
GO

输出:

开始
@Param1 is missing
1
50000
Msg 8152, Level 16, State 4, Procedure TestErrorHandling, Line 30
String or binary data would be truncated.
The statement has been terminated.
after insert
@Param1 is missing
1
50000
-8

令我惊讶的是,该程序并没有因为“字符串或二进制数据将被截断”而崩溃。换句话说,之后的代码仍然运行。此外,catch 块内的 SQL 错误不会更改 error_message。

所以问题是 - 在“BEGIN CATCH”部分处理意外错误的最佳实践是什么?我应该有另一个嵌套的 TRY/CATCH 吗?

注意:这是我最初的问题,但我认为将这个问题放在那里会使其脱离主题并混淆问题。这个问题得到回答后,我会回去更新那个: T-SQL Clear Errors

第 2 部分 - 稍后添加:
   DECLARE @returnStatus nvarchar(15);
Exec @returnStatus = TestErrorHandling
print @returnStatus

这将返回 -8。 -8 到底从何而来?

我也在尝试添加“返回”与“返回 0”。当 BizTalk 调用存储过程时,我希望他认为它成功完成,以免每 5 分钟重试 3 次。

补充:我认为这主要是我正在寻找的答案:
SQL try-catch statement not handling error (SQL Server 2008)

但它没有讨论我在这个问题中提出的最佳实践问题。

更新:这是演示艾伦响应的程序:
ALTER PROCEDURE TestErrorHandling2
@Param1 varchar(1) = ''
AS
BEGIN
/*
Unit test:
DECLARE @returnStatus nvarchar(15);
Exec @returnStatus = TestErrorHandling2
print @returnStatus

This is proof of concept on error handling.
See question: https://stackoverflow.com/questions/20245900/t-sql-is-error-handling-totally-turned-off-in-a-begin-catch-block
Testing to see if Truncation error stops or not.

*/

print 'Start'

INSERT INTO [FlightOrderUploadFailedLog]
(EvtTyp)
VALUES ('max size of EvtType is only varchar(15) so this should cause truncation')
print 'after insert #1'
print 'Message=' + IsNull(error_message(),'null')
print 'State=' + IsNull(convert(varchar(4),error_state()),'null')
print 'ErrorNumber=' + IsNull(convert(varchar(8),error_number()),'null')


BEGIN TRY
print 'Start - Begin Try'

INSERT INTO [FlightOrderUploadFailedLog]
(EvtTyp)
VALUES ('max size of EvtType is only varchar(15) so this should cause truncation')
print 'after insert #2 '
print 'Message=' + IsNull(error_message(),'null')
print 'State=' + IsNull(convert(varchar(4),error_state()),'null')
print 'ErrorNumber=' + IsNull(convert(varchar(8),error_number()),'null')


print 'The End'

END TRY
BEGIN CATCH
print 'Catch'
print 'Message=' + error_message()
print 'State=' + convert(varchar(4),error_state())
print 'ErrorNumber=' + convert(varchar(8),error_number())
return -- error has been theoretically handled by writing it to a database

END CATCH

END
GO

结果:
Start
Msg 8152, Level 16, State 4, Procedure TestErrorHandling2, Line 21
String or binary data would be truncated.
The statement has been terminated.
after insert #1
Message=null
State=null
ErrorNumber=null
Start - Begin Try

(0 row(s) affected)
Catch
Message=String or binary data would be truncated.
State=4
ErrorNumber=8152
-6

最佳答案

好吧,让我尽力而为。

“最佳实践”——我讨厌这句话,因为 10 次中有 9 次它纯粹是主观的,因为有人在某个地方读过一些东西。

所以我的主观答案是,看到你可以嵌套 try/catch 并放置一个 try/catch within the catch block - 那么我认为在 catch 中放置一个 try/catch 是很好的错误处理 - 如果你有什么可以抛出错误并且足够严重以至于你希望自己处理它。

所以 IMO - 最佳实践是,嵌套 try/catch 以获得更好的错误处理。

其次 - 您的“捕获”没有爆炸的原因是截断错误不是批处理终止错误。它的错误级别不够高,因此将执行后续语句。
只需使用打印语句尝试一下:

PRINT 'something'
--do your insert here
PRINT 'somethingelse'

然后你会看到你应该得到两个打印语句。
如果您愿意,您甚至可以通过将 ANSI_WARNINGS 更改为 OFF 来抑制截断错误。不是我会推荐,但好吧...... :)

如果您的 catch 中有 try/catch,那应该会捕获您的 truncate 错误,因为严重性足以触发 catch。

关于sql-server-2008 - T-SQL 是否在 "BEGIN CATCH" block 中完全关闭了错误处理?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20245900/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com