如何从Windows命令行获取应用程序退出代码?

Skr*_*rud 764 windows command-line cmd process exit-code

我正在运行一个程序,想看看它的返回代码是什么(因为它根据不同的错误返回不同的代码).

我知道在Bash中我可以通过运行来做到这一点

回声$?

在Windows上使用cmd.exe时该怎么办?

DrF*_*yd5 930

名为errorlevel存储退出代码的伪环境变量:

echo Exit Code is %errorlevel%
Run Code Online (Sandbox Code Playgroud)

此外,该if命令具有特殊语法:

if errorlevel
Run Code Online (Sandbox Code Playgroud)

详情if /?请见.

@echo off
my_nify_exe.exe
if errorlevel 1 (
   echo Failure Reason Given is %errorlevel%
   exit /b %errorlevel%
)
Run Code Online (Sandbox Code Playgroud)

警告:如果设置环境变量名称errorlevel,%errorlevel%将返回该值而不是退出代码.使用(set errorlevel=)清除环境变量,允许errorlevel通过%errorlevel%环境变量访问真值.

  • 如果您直接从Windows命令行运行并且始终看到0返回,请参阅Gary的回答:http://stackoverflow.com/a/11476681/31629 (35认同)
  • 注意:如果errorlevel> = 1,则"errorlevel 1"为true.因此"errorlevel 0"将匹配所有内容.看看 /?".相反,您可以使用"if%ERRORLEVEL%EQU 0(..)". (10认同)
  • 此外,如果您在powershell中,您可以使用`echo Exit Code是$ LastExitCode` (9认同)
  • 发现即使发生错误,“%ERRORLEVEL%”仍为 0 的情况。在 cmd 文件中检查 `%ERRORLEVEL%` 时发生。尝试 `start /wait` 没有用。唯一有效的是`if errorlevel 1 (...)` (2认同)
  • 友情提示:%ErrorLevel% 是一个shell 变量,不是环境变量,它也返回一个`string` 而不是`int`,意味着你不能有效地使用`EQ`/`NEQ`。 (2认同)

Gar*_*ary 255

测试ErrorLevel适用于控制台应用程序,但正如dmihailescu暗示的那样,如果您尝试从命令提示符运行窗口化应用程序(例如基于Win32),则无法运行.窗口化应用程序将在后台运行,控件将立即返回到命令提示符(很可能ErrorLevel为零,表示该过程已成功创建).当窗口化应用程序最终退出时,其退出状态将丢失.

但是,更简单的替代方法是使用命令提示符的START /WAIT命令启动窗口化应用程序,而不是使用其他地方提到的基于控制台的C++启动程序.这将启动窗口化应用程序,等待它退出,然后将控制权返回到命令提示符,并设置进程的退出状态ErrorLevel.

start /wait something.exe
echo %errorlevel%
Run Code Online (Sandbox Code Playgroud)

  • 非常感谢"START /等待"的想法.这对我有用:) (19认同)
  • 它可能不起作用(始终为零)的另一个原因是当它位于“if”或“for”内部时。考虑使用 `!errorlevel!` 代替,如[本答案中所述](http://stackoverflow.com/a/4368104/33080)。 (3认同)
  • 很好的抓住.我不知道那个命令.我刚刚看到它工作>启动/等待notepad.exe (2认同)

Ada*_*eld 96

使用内置的ERRORLEVEL变量:

echo %ERRORLEVEL%
Run Code Online (Sandbox Code Playgroud)

要注意应用程序是否定义了名为ERRORLEVEL的环境变量!

  • @SteelBrain:它在PowerShell中称为`$ LastExitCode`. (13认同)
  • 它不是一个实际的环境变量(显然,如果*是*以这种方式命名的变量,它就会停止工作). (6认同)
  • 请注意它在powershell中不起作用 (3认同)

Cur*_*lop 20

如果要精确匹配错误代码(例如等于0),请使用:

@echo off
my_nify_exe.exe
if %ERRORLEVEL% EQU 0 (
   echo Success
) else (
   echo Failure Reason Given is %errorlevel%
   exit /b %errorlevel%
)
Run Code Online (Sandbox Code Playgroud)

if errorlevel 0匹配errorlevel> = 0.见if /?.

  • 不。无论情况如何,变量、命令(包括“if”)和“equ”都可以工作。 (2认同)

dmi*_*scu 14

使用未附加到控制台的程序时,它可能无法正常工作,因为当您认为有退出代码时,该应用程序可能仍在运行.用C++实现它的解决方案如下所示:

#include "stdafx.h"
#include "windows.h"
#include "stdio.h"
#include "tchar.h"
#include "stdio.h"
#include "shellapi.h"

int _tmain( int argc, TCHAR *argv[] )
{

    CString cmdline(GetCommandLineW());
    cmdline.TrimLeft('\"');
    CString self(argv[0]);
    self.Trim('\"');
    CString args = cmdline.Mid(self.GetLength()+1);
    args.TrimLeft(_T("\" "));
    printf("Arguments passed: '%ws'\n",args);
    STARTUPINFO si;
    PROCESS_INFORMATION pi;

    ZeroMemory( &si, sizeof(si) );
    si.cb = sizeof(si);
    ZeroMemory( &pi, sizeof(pi) );

    if( argc < 2 )
    {
        printf("Usage: %s arg1,arg2....\n", argv[0]);
        return -1;
    }

    CString strCmd(args);
    // Start the child process. 
    if( !CreateProcess( NULL,   // No module name (use command line)
        (LPTSTR)(strCmd.GetString()),        // Command line
        NULL,           // Process handle not inheritable
        NULL,           // Thread handle not inheritable
        FALSE,          // Set handle inheritance to FALSE
        0,              // No creation flags
        NULL,           // Use parent's environment block
        NULL,           // Use parent's starting directory 
        &si,            // Pointer to STARTUPINFO structure
        &pi )           // Pointer to PROCESS_INFORMATION structure
    ) 
    {
        printf( "CreateProcess failed (%d)\n", GetLastError() );
        return GetLastError();
    }
    else
        printf( "Waiting for \"%ws\" to exit.....\n", strCmd );

    // Wait until child process exits.
    WaitForSingleObject( pi.hProcess, INFINITE );
    int result = -1;
    if(!GetExitCodeProcess(pi.hProcess,(LPDWORD)&result))
    { 
        printf("GetExitCodeProcess() failed (%d)\n", GetLastError() );
    }
    else
        printf("The exit code for '%ws' is %d\n",(LPTSTR)(strCmd.GetString()), result );
    // Close process and thread handles. 
    CloseHandle( pi.hProcess );
    CloseHandle( pi.hThread );
    return result;
}
Run Code Online (Sandbox Code Playgroud)


Roc*_*tor 7

值得注意的是.BAT和.CMD文件的操作不同。

阅读https://ss64.com/nt/errorlevel.html时,请注意以下几点:

.CMD和.BAT批处理文件设置错误级别的方式之间有一个关键区别:

运行“新”内部命令:APPEND,ASSOC,PATH,PROMPT,FTYPE和SET的旧的.BAT批处理脚本仅在发生错误时设置ERRORLEVEL。因此,如果批处理脚本中有两个命令,而第一个命令失败,则即使第二个命令成功后,ERRORLEVEL仍将保持设置状态。

这会使调试问题的BAT脚本更加困难,CMD批处理脚本更加一致,并且在您运行[source]的每个命令后都会设置ERRORLEVEL。

当我执行连续的命令时,这没有使我感到悲伤,但是即使发生故障,ERRORLEVEL仍然保持不变。