ฉันจะรับรหัสออกจากแอปพลิเคชันจากบรรทัดคำสั่ง Windows ได้อย่างไร


797

ฉันกำลังเรียกใช้โปรแกรมและต้องการดูว่ารหัสส่งคืนของมันคืออะไร

ฉันรู้ใน Bash ฉันสามารถทำได้โดยใช้

echo $

ฉันต้องทำอย่างไรเมื่อใช้ cmd.exe บน Windows



1
Googled สำหรับ "Win8 วิธีรับพร้อมท์ CMD เพื่อแสดงสถานะการออก" อย่างที่เราสามารถทำได้ใน Linux นี่คือตัวเลือกอันดับต้น ๆ และมีความแม่นยำ
SDsolar

1
คุณสามารถดูได้อย่างรวดเร็วว่าแอปส่งคืนอะไร:app.exe & echo %errorlevel%
marbel82

คำตอบ:


976

ตัวแปรสภาพแวดล้อมหลอกชื่อที่errorlevelเก็บรหัสทางออก:

echo Exit Code is %errorlevel%

นอกจากนี้ifคำสั่งยังมีไวยากรณ์พิเศษ:

if errorlevel

ดูif /?รายละเอียดที่

ตัวอย่าง

@echo off
my_nify_exe.exe
if errorlevel 1 (
   echo Failure Reason Given is %errorlevel%
   exit /b %errorlevel%
)

คำเตือน: ถ้าคุณตั้งชื่อตัวแปรสภาพแวดล้อมerrorlevel, %errorlevel%จะกลับมาที่คุ้มค่าและไม่รหัสทางออก ใช้ ( set errorlevel=) เพื่อล้างตัวแปรสภาพแวดล้อมทำให้สามารถเข้าถึงค่าที่แท้จริงของerrorlevelผ่าน%errorlevel%ตัวแปรสภาพแวดล้อม


38
หากคุณกำลังเรียกใช้โดยตรงจากบรรทัดคำสั่งของ Windows และเห็น 0 คืนเสมอให้ดูคำตอบของ Gary: stackoverflow.com/a/11476681/31629
Ken

9
นอกจากนี้หากคุณอยู่ใน PowerShell คุณสามารถใช้echo Exit Code is $LastExitCode
แบรนดอน Pugh

11
หมายเหตุ: "errorlevel 1" เป็นจริงถ้า errorlevel> = 1 ดังนั้น "errorlevel 0" จะจับคู่ทุกอย่าง ดู "if /?" คุณสามารถใช้ "ถ้า% ERRORLEVEL% EQU 0 (.. )" แทน
เคอร์ติส Yallop

1
พบกรณีที่%ERRORLEVEL%เป็น 0 แม้ว่าจะมีข้อผิดพลาดเกิดขึ้น เกิดขึ้นเมื่อตรวจสอบ%ERRORLEVEL%ในไฟล์ cmd ความพยายามstart /waitไม่ได้ผล สิ่งเดียวที่ทำงานเป็นif errorlevel 1 (...)
AlikElzin-kilaka

1
คำแนะนำที่เป็นมิตร:% ErrorLevel% เป็นตัวแปรเชลล์ไม่ใช่ตัวแปรสภาพแวดล้อมและยังส่งคืนค่าstringไม่ใช่intหมายถึงคุณไม่สามารถใช้EQ/ ได้NEQอย่างมีประสิทธิภาพ
kayleeFrye_onDeck

277

การทดสอบใช้ErrorLevelงานได้กับแอปพลิเคชันคอนโซลแต่ตามที่dmihailescu บอกไว้นี่จะไม่ทำงานหากคุณพยายามเรียกใช้แอปพลิเคชันแบบมีหน้าต่าง (เช่นที่ใช้ Win32) จากพรอมต์คำสั่ง แอปพลิเคชันที่มีหน้าต่างจะทำงานในพื้นหลังและการควบคุมจะกลับไปที่พรอมต์คำสั่งทันที (ส่วนใหญ่มีค่าเป็นErrorLevelศูนย์เพื่อระบุว่ากระบวนการสร้างเสร็จเรียบร้อยแล้ว) เมื่อแอปพลิเคชันที่มีหน้าต่างออกในที่สุดสถานะการออกจะหายไป

แทนที่จะใช้ตัวเรียกใช้งาน C ++ บนคอนโซลที่กล่าวถึงที่อื่น แต่ทางเลือกที่ง่ายกว่าคือการเริ่มต้นแอปพลิเคชันแบบมีหน้าต่างโดยใช้คำสั่งของพรอมต์START /WAITคำสั่ง ErrorLevelนี้จะเริ่มการทำงานของแอพลิเคชันหน้าต่างรอให้มันออกแล้วส่งคืนการควบคุมคำสั่งพรอมต์ที่มีสถานะออกจากชุดกระบวนการ

start /wait something.exe
echo %errorlevel%

20
ขอบคุณมากสำหรับแนวคิด "เริ่ม / รอ" ที่ทำงานให้ฉัน :)
Timotei

3
รับได้สวย. ฉันไม่รู้เกี่ยวกับคำสั่งนั้น ฉันเพิ่งเห็นว่ามันใช้งานได้> start / wait notepad.exe
dmihailescu

1
อีกเหตุผลหนึ่งที่ว่าทำไมมันอาจจะไม่ทำงาน (เสมอศูนย์) คือเมื่อมันเป็นภายในหรือif forลองใช้!errorlevel!แทนตามที่อธิบายไว้ในคำตอบนี้
Roman Starkov

100

6
มันไม่ได้เป็นตัวแปรสภาพแวดล้อมที่เกิดขึ้นจริง (ซึ่งเห็นได้ชัดว่าทำไมมันสิ้นสุดสภาพการทำงานถ้ามีเป็นตัวแปรชื่อวิธีการที่)
Joey

17
@SteelBrain: มันถูกเรียก$LastExitCodeใน PowerShell
Alex A.

23

หากคุณต้องการจับคู่รหัสข้อผิดพลาดอย่างแน่นอน (เช่นเท่ากับ 0) ให้ใช้สิ่งนี้:

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

if errorlevel 0การแข่งขันerrorlevel> = 0 if /?ดู


เป็นกรณี ๆ ไปหรือไม่?
Nishant

1
ไม่ vars, คำสั่ง (รวมถึง "ถ้า") และ "equ" ทำงานไม่ว่าในกรณีใด
เคอร์ติส Yallop

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;
}

ในการกำหนดค่าบางอย่างคุณควรเพิ่ม #include <atlstr.h> เพื่อให้ประเภท CString ถูกนำมาใช้ใหม่
Jake OPJ

8

เป็นที่น่าสังเกตว่าไฟล์. BAT และ. CMD นั้นทำงานแตกต่างกัน

การอ่านhttps://ss64.com/nt/errorlevel.htmlมันบันทึกสิ่งต่อไปนี้:

มีความแตกต่างที่สำคัญระหว่างวิธีการที่ชุดไฟล์. CMD และ. BAT ชุดข้อผิดพลาดระดับ:

สคริปต์แบตช์. BAT แบบเก่าที่รันคำสั่งภายใน 'ใหม่': APPEND, ASSOC, PATH, PROMPT, FTYPE และ SET จะตั้งค่า ERRORLEVEL หากเกิดข้อผิดพลาดเท่านั้น ดังนั้นหากคุณมีสองคำสั่งในสคริปต์ชุดงานและคำสั่งแรกล้มเหลว ERRORLEVEL จะยังคงถูกตั้งค่าแม้หลังจากที่คำสั่งที่สองจะประสบความสำเร็จ

วิธีนี้จะทำให้การดีบักปัญหาสคริปต์ BAT นั้นยากขึ้นสคริปต์ชุดงาน CMD นั้นสอดคล้องกันมากขึ้นและจะตั้งค่า ERRORLEVEL หลังจากทุกคำสั่งที่คุณเรียกใช้ [แหล่งที่มา]

สิ่งนี้ทำให้ฉันไม่มีความโศกเศร้าในขณะที่ฉันกำลังดำเนินการคำสั่งต่อเนื่อง แต่ ERRORLEVEL จะยังคงไม่เปลี่ยนแปลงแม้ในกรณีที่เกิดความล้มเหลว


0

ณ จุดหนึ่งฉันจำเป็นต้องผลักดันบันทึกเหตุการณ์จาก Cygwin ไปยังบันทึกเหตุการณ์ของ Windows อย่างถูกต้อง ฉันต้องการให้ข้อความใน WEVL เป็นแบบกำหนดเองมีรหัสทางออกที่ถูกต้องรายละเอียดลำดับความสำคัญข้อความ ฯลฯ ดังนั้นฉันจึงสร้างสคริปต์ Bash เล็กน้อยเพื่อดูแลสิ่งนี้ นี่มันอยู่บน GitHub, logit.sh

ข้อความที่ตัดตอนมาบางส่วน:

usage: logit.sh [-h] [-p] [-i=n] [-s] <description>
example: logit.sh -p error -i 501 -s myscript.sh "failed to run the mount command"

นี่คือส่วนเนื้อหาไฟล์ชั่วคราว:

LGT_TEMP_FILE="$(mktemp --suffix .cmd)"
cat<<EOF>$LGT_TEMP_FILE
    @echo off
    set LGT_EXITCODE="$LGT_ID"
    exit /b %LGT_ID%
EOF
unix2dos "$LGT_TEMP_FILE"

นี่คือฟังก์ชั่นเพื่อสร้างกิจกรรมใน WEVL:

__create_event () {
    local cmd="eventcreate /ID $LGT_ID /L Application /SO $LGT_SOURCE /T $LGT_PRIORITY /D "
    if [[ "$1" == *';'* ]]; then
        local IFS=';'
        for i in "$1"; do
            $cmd "$i" &>/dev/null
        done
    else
        $cmd "$LGT_DESC" &>/dev/null
    fi
}

การดำเนินการแบตช์สคริปต์และเรียกใช้ __create_event:

cmd /c "$(cygpath -wa "$LGT_TEMP_FILE")"
__create_event
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.