ฉันกำลังเรียกใช้โปรแกรมและต้องการดูว่ารหัสส่งคืนของมันคืออะไร
ฉันรู้ใน Bash ฉันสามารถทำได้โดยใช้
echo $
ฉันต้องทำอย่างไรเมื่อใช้ cmd.exe บน Windows
app.exe & echo %errorlevel%
ฉันกำลังเรียกใช้โปรแกรมและต้องการดูว่ารหัสส่งคืนของมันคืออะไร
ฉันรู้ใน Bash ฉันสามารถทำได้โดยใช้
echo $
ฉันต้องทำอย่างไรเมื่อใช้ cmd.exe บน Windows
app.exe & echo %errorlevel%
คำตอบ:
ตัวแปรสภาพแวดล้อมหลอกชื่อที่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%
ตัวแปรสภาพแวดล้อม
echo Exit Code is $LastExitCode
%ERRORLEVEL%
เป็น 0 แม้ว่าจะมีข้อผิดพลาดเกิดขึ้น เกิดขึ้นเมื่อตรวจสอบ%ERRORLEVEL%
ในไฟล์ cmd ความพยายามstart /wait
ไม่ได้ผล สิ่งเดียวที่ทำงานเป็นif errorlevel 1 (...)
string
ไม่ใช่int
หมายถึงคุณไม่สามารถใช้EQ
/ ได้NEQ
อย่างมีประสิทธิภาพ
การทดสอบใช้ErrorLevel
งานได้กับแอปพลิเคชันคอนโซลแต่ตามที่dmihailescu บอกไว้นี่จะไม่ทำงานหากคุณพยายามเรียกใช้แอปพลิเคชันแบบมีหน้าต่าง (เช่นที่ใช้ Win32) จากพรอมต์คำสั่ง แอปพลิเคชันที่มีหน้าต่างจะทำงานในพื้นหลังและการควบคุมจะกลับไปที่พรอมต์คำสั่งทันที (ส่วนใหญ่มีค่าเป็นErrorLevel
ศูนย์เพื่อระบุว่ากระบวนการสร้างเสร็จเรียบร้อยแล้ว) เมื่อแอปพลิเคชันที่มีหน้าต่างออกในที่สุดสถานะการออกจะหายไป
แทนที่จะใช้ตัวเรียกใช้งาน C ++ บนคอนโซลที่กล่าวถึงที่อื่น แต่ทางเลือกที่ง่ายกว่าคือการเริ่มต้นแอปพลิเคชันแบบมีหน้าต่างโดยใช้คำสั่งของพรอมต์START /WAIT
คำสั่ง ErrorLevel
นี้จะเริ่มการทำงานของแอพลิเคชันหน้าต่างรอให้มันออกแล้วส่งคืนการควบคุมคำสั่งพรอมต์ที่มีสถานะออกจากชุดกระบวนการ
start /wait something.exe
echo %errorlevel%
if
for
ลองใช้!errorlevel!
แทนตามที่อธิบายไว้ในคำตอบนี้
ใช้ตัวแปร ERRORLEVEL ในตัว:
echo %ERRORLEVEL%
แต่ระวังหากแอปพลิเคชันได้กำหนดตัวแปรสภาพแวดล้อมชื่อว่า ERRORLEVEL !
$LastExitCode
ใน PowerShell
หากคุณต้องการจับคู่รหัสข้อผิดพลาดอย่างแน่นอน (เช่นเท่ากับ 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 /?
ดู
อาจทำงานไม่ถูกต้องเมื่อใช้โปรแกรมที่ไม่ได้ต่ออยู่กับคอนโซลเนื่องจากแอปนั้นอาจยังทำงานอยู่ในขณะที่คุณคิดว่าคุณมีรหัสออก วิธีแก้ปัญหาใน 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;
}
เป็นที่น่าสังเกตว่าไฟล์. BAT และ. CMD นั้นทำงานแตกต่างกัน
การอ่านhttps://ss64.com/nt/errorlevel.htmlมันบันทึกสิ่งต่อไปนี้:
มีความแตกต่างที่สำคัญระหว่างวิธีการที่ชุดไฟล์. CMD และ. BAT ชุดข้อผิดพลาดระดับ:
สคริปต์แบตช์. BAT แบบเก่าที่รันคำสั่งภายใน 'ใหม่': APPEND, ASSOC, PATH, PROMPT, FTYPE และ SET จะตั้งค่า ERRORLEVEL หากเกิดข้อผิดพลาดเท่านั้น ดังนั้นหากคุณมีสองคำสั่งในสคริปต์ชุดงานและคำสั่งแรกล้มเหลว ERRORLEVEL จะยังคงถูกตั้งค่าแม้หลังจากที่คำสั่งที่สองจะประสบความสำเร็จ
วิธีนี้จะทำให้การดีบักปัญหาสคริปต์ BAT นั้นยากขึ้นสคริปต์ชุดงาน CMD นั้นสอดคล้องกันมากขึ้นและจะตั้งค่า ERRORLEVEL หลังจากทุกคำสั่งที่คุณเรียกใช้ [แหล่งที่มา]
สิ่งนี้ทำให้ฉันไม่มีความโศกเศร้าในขณะที่ฉันกำลังดำเนินการคำสั่งต่อเนื่อง แต่ ERRORLEVEL จะยังคงไม่เปลี่ยนแปลงแม้ในกรณีที่เกิดความล้มเหลว
ณ จุดหนึ่งฉันจำเป็นต้องผลักดันบันทึกเหตุการณ์จาก 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