364

有没有办法立即停止 SQL 服务器中 SQL 脚本的执行,比如“break”或“exit”命令?

我有一个脚本在开始插入之前进行一些验证和查找,如果任何验证或查找失败,我希望它停止。

4

21 回答 21

416

raiseerror方法_

raiserror('Oh no a fatal error', 20, -1) with log

这将终止连接,从而停止脚本的其余部分运行。

请注意,WITH LOG要以这种方式工作,需要 20 或更高的严重性级别和选项。

这甚至适用于 GO 语句,例如。

print 'hi'
go
raiserror('Oh no a fatal error', 20, -1) with log
go
print 'ho'

会给你输出:

hi
Msg 2745, Level 16, State 2, Line 1
Process ID 51 has raised user error 50000, severity 20. SQL Server is terminating this process.
Msg 50000, Level 20, State 1, Line 1
Oh no a fatal error
Msg 0, Level 20, State 0, Line 0
A severe error occurred on the current command.  The results, if any, should be discarded.

请注意,未打印“ho”。

注意事项:

  • 这仅在您以 admin('sysadmin' 角色)身份登录时有效,并且也使您没有数据库连接。
  • 如果您不是以管理员身份登录,则 RAISEERROR() 调用本身将失败,脚本将继续执行
  • 使用 sqlcmd.exe 调用时,将报告退出代码 2745。

参考:http ://www.mydatabasesupport.com/forums/ms-sqlserver/174037-sql-server-2000-abort-whole-script.html#post761334

noexec 方法

另一种适用于 GO 语句的方法是set noexec on. 这会导致脚本的其余部分被跳过。它不会终止连接,但您需要noexec在执行任何命令之前再次关闭。

例子:

print 'hi'
go

print 'Fatal error, script will not continue!'
set noexec on

print 'ho'
go

-- last line of the script
set noexec off -- Turn execution back on; only needed in SSMS, so as to be able 
               -- to run this script again in the same session.
于 2009-04-29T23:43:23.703 回答
201

只需使用 RETURN (它可以在存储过程的内部和外部工作)。

于 2009-03-18T17:23:19.433 回答
59

如果可以使用 SQLCMD 模式,那么咒语

:on error exit

(包括冒号)将导致 RAISERROR 实际停止脚本。例如,

:on error exit

IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[SOMETABLE]') AND type in (N'U')) 
    RaisError ('This is not a Valid Instance Database', 15, 10)
GO

print 'Keep Working'

将输出:

Msg 50000, Level 15, State 10, Line 3
This is not a Valid Instance Database
** An error was encountered during execution of batch. Exiting.

批次将停止。如果没有打开 SQLCMD 模式,你会得到关于冒号的解析错误。不幸的是,它并不是完全防弹的,就好像脚本在没有 SQLCMD 模式的情况下运行一样,SQL Managment Studio 甚至可以轻松克服解析时间错误!不过,如果您从命令行运行它们,这很好。

于 2010-04-07T06:13:39.770 回答
21

我不会使用 RAISERROR——SQL 有可用于此目的的 IF 语句。进行验证和查找并设置局部变量,然后使用 IF 语句中的变量值使插入有条件。

您不需要检查每个验证测试的可变结果。您通常可以只使用一个标志变量来确认所有条件都已通过:

declare @valid bit

set @valid = 1

if -- Condition(s)
begin
  print 'Condition(s) failed.'
  set @valid = 0
end

-- Additional validation with similar structure

-- Final check that validation passed
if @valid = 1
begin
  print 'Validation succeeded.'

  -- Do work
end

即使您的验证更复杂,您也应该只需要在最终检查中包含几个标志变量。

于 2009-03-18T17:07:12.763 回答
17

在 SQL 2012+ 中,您可以使用THROW

THROW 51000, 'Stopping execution because validation failed.', 0;
PRINT 'Still Executing'; -- This doesn't execute with THROW

来自 MSDN:

引发异常并将执行转移到 TRY...CATCH 构造的 CATCH 块...如果 TRY...CATCH 构造不可用,则结束会话。设置引发异常的行号和过程。严重性设置为 16。

于 2015-11-19T16:21:15.520 回答
14

我使用事务成功扩展了 noexec 开/关解决方案,以全有或全无的方式运行脚本。

set noexec off

begin transaction
go

<First batch, do something here>
go
if @@error != 0 set noexec on;

<Second batch, do something here>
go
if @@error != 0 set noexec on;

<... etc>

declare @finished bit;
set @finished = 1;

SET noexec off;

IF @finished = 1
BEGIN
    PRINT 'Committing changes'
    COMMIT TRANSACTION
END
ELSE
BEGIN
    PRINT 'Errors occured. Rolling back changes'
    ROLLBACK TRANSACTION
END

显然编译器“理解” IF 中的@finished 变量,即使出现错误并且执行被禁用。但是,仅当未禁用执行时,该值才设置为 1。因此,我可以相应地很好地提交或回滚事务。

于 2011-10-25T12:27:31.450 回答
12

您可以将 SQL 语句包装在 WHILE 循环中,并在需要时使用 BREAK

WHILE 1 = 1
BEGIN
   -- Do work here
   -- If you need to stop execution then use a BREAK


    BREAK; --Make sure to have this break at the end to prevent infinite loop
END
于 2009-03-18T17:27:14.947 回答
12

您可以使用GOTO语句更改执行流程:

IF @ValidationResult = 0
BEGIN
    PRINT 'Validation fault.'
    GOTO EndScript
END

/* our code */

EndScript:
于 2016-12-09T15:21:25.497 回答
11

进一步 refinig Sglasses 方法,上面的行强制使用 SQLCMD 模式,如果不使用 SQLCMD 模式,则终止脚本,或者使用:on error exit在任何错误时退出
CONTEXT_INFO用于跟踪状态。

SET CONTEXT_INFO  0x1 --Just to make sure everything's ok
GO 
--treminate the script on any error. (Requires SQLCMD mode)
:on error exit 
--If not in SQLCMD mode the above line will generate an error, so the next line won't hit
SET CONTEXT_INFO 0x2
GO
--make sure to use SQLCMD mode ( :on error needs that)
IF CONTEXT_INFO()<>0x2 
BEGIN
    SELECT CONTEXT_INFO()
    SELECT 'This script must be run in SQLCMD mode! (To enable it go to (Management Studio) Query->SQLCMD mode)\nPlease abort the script!'
    RAISERROR('This script must be run in SQLCMD mode! (To enable it go to (Management Studio) Query->SQLCMD mode)\nPlease abort the script!',16,1) WITH NOWAIT 
    WAITFOR DELAY '02:00'; --wait for the user to read the message, and terminate the script manually
END
GO

----------------------------------------------------------------------------------
----THE ACTUAL SCRIPT BEGINS HERE-------------
于 2010-08-24T14:02:13.647 回答
8

这是一个存储过程吗?如果是这样,我认为您可以只做一个返回,例如“返回NULL”;

于 2009-03-18T17:07:52.870 回答
7

我建议您将适当的代码块包装在 try catch 块中。然后,您可以使用严重性为 11 的 Raiserror 事件,以便根据需要中断到 catch 块。如果您只想引发错误但在 try 块中继续执行,则使用较低的严重性。

有道理?

干杯,约翰

[编辑以包括 BOL 参考]

http://msdn.microsoft.com/en-us/library/ms175976(SQL.90).aspx

于 2009-03-18T17:18:54.997 回答
7

RETURN一直在这里使用,在脚本中工作或Stored Procedure

ROLLBACK如果你在其中,请确保你的事务,否则RETURN立即会导致一个未提交的未提交事务

于 2009-03-18T18:28:46.950 回答
5

这些都不适用于“GO”语句。在此代码中,无论严重性是 10 还是 11,您都会得到最终的 PRINT 语句。

测试脚本:

-- =================================
PRINT 'Start Test 1 - RAISERROR'

IF 1 = 1 BEGIN
    RAISERROR('Error 1, level 11', 11, 1)
    RETURN
END

IF 1 = 1 BEGIN
    RAISERROR('Error 2, level 11', 11, 1)
    RETURN
END
GO

PRINT 'Test 1 - After GO'
GO

-- =================================
PRINT 'Start Test 2 - Try/Catch'

BEGIN TRY
    SELECT (1 / 0) AS CauseError
END TRY
BEGIN CATCH
    SELECT ERROR_MESSAGE() AS ErrorMessage
    RAISERROR('Error in TRY, level 11', 11, 1)
    RETURN
END CATCH
GO

PRINT 'Test 2 - After GO'
GO

结果:

Start Test 1 - RAISERROR
Msg 50000, Level 11, State 1, Line 5
Error 1, level 11
Test 1 - After GO
Start Test 2 - Try/Catch
 CauseError
-----------

ErrorMessage
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
Divide by zero error encountered.

Msg 50000, Level 11, State 1, Line 10
Error in TRY, level 11
Test 2 - After GO

完成这项工作的唯一方法是编写不带GO语句的脚本。有时这很容易。有时这是相当困难的。(使用类似的东西IF @error <> 0 BEGIN ...。)

于 2009-03-18T21:10:16.127 回答
4

你可以使用RAISERROR

于 2009-03-18T17:05:27.930 回答
3

这是我的解决方案:

...

BEGIN
    raiserror('Invalid database', 15, 10)
    rollback transaction
    return
END
于 2012-06-27T08:40:44.163 回答
3

您可以使用 GOTO 语句。试试这个。这对你有用。

WHILE(@N <= @Count)
BEGIN
    GOTO FinalStateMent;
END

FinalStatement:
     Select @CoumnName from TableName
于 2015-09-07T06:58:18.527 回答
1

谢谢你的答案!

raiserror()工作正常,但您不应该忘记该return语句,否则脚本将继续没有错误!(因此 raiserror 不是“throwerror” ;-))当然,如果需要,还可以进行回滚!

raiserror()很高兴告诉执行脚本的人出了点问题。

于 2009-03-31T13:52:18.187 回答
1

如果您只是在 Management Studio 中执行脚本,并且想要在第一个错误时停止执行或回滚事务(如果使用),那么我认为最好的方法是使用 try catch 块(SQL 2005 及更高版本)。如果您正在执行脚本文件,这在 Management Studio 中效果很好。存储过程也可以始终使用它。

于 2012-07-05T07:34:31.370 回答
1

将其包含在 try catch 块中,然后执行将转移到 catch。

BEGIN TRY
    PRINT 'This will be printed'
    RAISERROR ('Custom Exception', 16, 1);
    PRINT 'This will not be printed'
END TRY
BEGIN CATCH
    PRINT 'This will be printed 2nd'
END CATCH;
于 2020-05-28T11:34:12.163 回答
0

回到过去,我们使用以下...效果最好:

RAISERROR ('Error! Connection dead', 20, 127) WITH LOG
于 2019-01-03T22:59:16.023 回答
0

非常感谢这里的所有其他人和我读过的其他帖子。但是在@jaraics 回答之前,没有什么能满足我的所有需求。

我见过的大多数答案都忽略了多批次的脚本。他们忽略了 SSMS 和 SQLCMD 中的双重用法。我的脚本在 SSMS 中完全可以运行——但我想要 F5 预防,这样他们就不会意外删除现有的一组对象。

SET PARSEONLY ON工作得很好,可以防止不需要的 F5。但是你不能用 SQLCMD 运行。

另一件让我慢了一段时间的事情是,当出现错误时,批处理将如何跳过任何进一步的命令——所以我SET NOCOUNT ON被跳过了,因此脚本仍然运行。

无论如何,我稍微修改了 jaraics 的答案:(在这种情况下,我还需要一个数据库才能从命令行激活)

-----------------------------------------------------------------------
-- Prevent accidental F5
-- Options:
--     1) Highlight everything below here to run
--     2) Disable this safety guard
--     3) or use SQLCMD
-----------------------------------------------------------------------
set NOEXEC OFF                             -- Reset in case it got stuck ON
set CONTEXT_INFO  0x1                      -- A 'variable' that can pass batch boundaries
GO                                         -- important !
if $(SQLCMDDBNAME) is not null
    set CONTEXT_INFO 0x2                   -- If above line worked, we're in SQLCMD mode
GO                                         -- important !
if CONTEXT_INFO()<>0x2 
begin
    select 'F5 Pressed accidentally.'
    SET NOEXEC ON                          -- skip rest of script
END
GO                                         -- important !
-----------------------------------------------------------------------

< rest of script . . . . . >


GO
SET NOEXEC OFF
print 'DONE'
于 2021-04-28T23:01:50.750 回答