如何使批处理文件在遇到错误时终止?

我有一个批处理文件,它用不同的参数反复调用相同的可执行文件。如果其中一个调用返回任何级别的错误代码,我如何使它立即终止?

基本上,我想要MSBuild的ContinueOnError=false的等效。

313625 次浏览

if语句中检查errorlevel,然后检查exit /b(仅退出batch文件,而不是整个cmd.exe进程)是否有除0以外的值。

same-executable-over-and-over.exe /with different "parameters"
if %errorlevel% neq 0 exit /b %errorlevel%

如果您希望errorlevel的值传播到批处理文件之外

if %errorlevel% neq 0 exit /b %errorlevel%

但如果这是在for中,就有点棘手了。你需要的是:

setlocal enabledelayedexpansion
for %%f in (C:\Windows\*) do (
same-executable-over-and-over.exe /with different "parameters"
if !errorlevel! neq 0 exit /b !errorlevel!
)

编辑:你必须在每个命令之后检查错误。在cmd.exe/command.com批处理中没有全局的“on error goto”类型的构造。我还根据CodeMonkey更新了我的代码,尽管我从未在XP或Vista上的任何批处理中遇到过负错误级别。

一个小更新,你应该改变检查“if errorlevel 1”为以下…

IF %ERRORLEVEL% NEQ 0

这是因为在XP上你可以得到负数作为错误。0 =没问题,其他的都是问题。

记住DOS处理“IF ERRORLEVEL”测试的方式。如果您正在检查的数字是该数字或更高,它将返回true,所以如果您正在寻找特定的错误数字,您需要从255开始,然后往下算。

在每行添加|| goto :label,然后定义一个:label

例如,创建这个.cmd文件:

@echo off


echo Starting very complicated batch file...
ping -invalid-arg || goto :error
echo OH noes, this shouldn't have succeeded.
goto :EOF


:error
echo Failed with error #%errorlevel%.
exit /b %errorlevel%

另见关于退出批处理文件子例程的问题

@echo off


set startbuild=%TIME%


C:\WINDOWS\Microsoft.NET\Framework\v3.5\msbuild.exe c:\link.xml /flp1:logfile=c:\link\errors.log;errorsonly /flp2:logfile=c:\link\warnings.log;warningsonly || goto :error


copy c:\app_offline.htm "\\lawpccnweb01\d$\websites\OperationsLinkWeb\app_offline.htm"


del \\lawpccnweb01\d$\websites\OperationsLinkWeb\bin\ /Q


echo Start Copy: %TIME%


set copystart=%TIME%


xcopy C:\link\_PublishedWebsites\OperationsLink \\lawpccnweb01\d$\websites\OperationsLinkWeb\ /s /y /d


del \\lawpccnweb01\d$\websites\OperationsLinkWeb\app_offline.htm


echo Started Build: %startbuild%
echo Started Copy: %copystart%
echo Finished Copy: %TIME%


c:\link\warnings.log


:error


c:\link\errors.log

最短的:

command || exit /b

如果需要,可以设置退出码:

command || exit /b 666

你也可以登录:

command || echo ERROR && exit /b

我们不能总是依赖ERRORLEVEL,因为很多时候外部程序或批处理脚本不返回退出码。

在这种情况下,我们可以像这样使用通用检查失败:

IF EXIST %outfile% (DEL /F %outfile%)
CALL some_script.bat -o %outfile%
IF NOT EXIST %outfile%  (ECHO ERROR & EXIT /b)

如果程序输出一些东西到控制台,我们也可以检查它。

some_program.exe 2>&1 | FIND "error message here" && (ECHO ERROR & EXIT /b)
some_program.exe 2>&1 | FIND "Done processing." || (ECHO ERROR & EXIT /b)

无论我如何尝试,错误级别总是保持0,即使msbuild失败。所以我建立了我的变通方案:

生成项目并将日志保存到Build.log

SET Build_Opt=/flp:summary;logfile=Build.log;append=true


msbuild "myproj.csproj" /t:rebuild /p:Configuration=release /fl %Build_Opt%

在构建日志中搜索“0错误”字符串,将结果设置为var

FOR /F "tokens=* USEBACKQ" %%F IN (`find /c /i "0 Error" Build.log`) DO (
SET var=%%F
)
echo %var%

获取最后一个字符,该字符表示搜索字符串包含多少行

set result=%var:~-1%


echo "%result%"

如果未找到字符串,则错误> 0,构建失败

if "%result%"=="0" ( echo "build failed" )

这个解决方案的灵感来自Mechaflash在如何在批处理文件中将命令输出设置为变量的帖子

https://ss64.com/nt/syntax-substring.html

下面是BASH和Windows CMD的通晓多种语言的程序,它运行一系列命令,如果其中任何一个命令失败,就退出:

#!/bin/bash 2> nul


:; set -o errexit
:; function goto() { return $?; }


command 1 || goto :error


command 2 || goto :error


command 3 || goto :error


:; exit 0
exit /b 0


:error
exit /b %errorlevel%

我在过去的一个多平台持续集成脚本中使用过这种类型的东西。

我更喜欢OR形式的命令,因为我发现它们最有可读性(如 而不是在每个命令后面加一个if)。然而,幼稚的方式 这样做,command || exit /b %ERRORLEVEL%错误的.

这是因为batch在第一次读取一行时展开变量 而不是被使用的时候。这意味着如果command在行中 上述失败,批处理文件正常退出,但退出时返回代码为0, 因为这是%ERRORLEVEL%在开始时的值 线。显然,这在我们的脚本中是不可取的,因此我们必须启用 推迟扩张,像这样:

SETLOCAL EnableDelayedExpansion


command-1 || exit /b !ERRORLEVEL!
command-2 || exit /b !ERRORLEVEL!
command-3 || exit /b !ERRORLEVEL!
command-4 || exit /b !ERRORLEVEL!
这个代码片段将执行命令1-4,如果其中任何一个失败,它将执行

.使用与失败命令相同的退出代码退出