私はプログラムを実行していて、その戻りコードが何であるかを確認したいと思います(さまざまなエラーに基づいてさまざまなコードを返すため)。
私はBashで実行してこれを行うことができることを知っています
エコー$?
Windowsでcmd.exeを使用する場合はどうすればよいですか?
app.exe & echo %errorlevel%
私はプログラムを実行していて、その戻りコードが何であるかを確認したいと思います(さまざまなエラーに基づいてさまざまなコードを返すため)。
私はBashで実行してこれを行うことができることを知っています
エコー$?
Windowsでcmd.exeを使用する場合はどうすればよいですか?
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
ません。また、notを返します。int
つまり、EQ
/をNEQ
効果的に使用できません。
テストErrorLevel
はコンソールアプリケーションで機能しますが、dmihailescuが示唆しているように、コマンドプロンプトからウィンドウアプリケーション(Win32ベースなど)を実行しようとすると、機能しません。ウィンドウ化されたアプリケーションはバックグラウンドで実行され、コントロールはすぐにコマンドプロンプトに戻ります(ErrorLevel
プロセスが正常に作成されたことを示すゼロの場合がほとんどです)。ウィンドウ化されたアプリケーションが最終的に終了すると、その終了ステータスは失われます。
ただし、他の場所で説明されているコンソールベースのC ++ランチャーを使用する代わりに、コマンドプロンプトのSTART /WAIT
コマンドを使用してウィンドウアプリケーションを起動する方法がより簡単です。これにより、ウィンドウ化されたアプリケーションが起動し、終了するのを待ってから、コマンドプロンプトに制御を戻し、プロセスの終了ステータスをに設定しErrorLevel
ます。
start /wait something.exe
echo %errorlevel%
エラーコードを正確に一致させたい場合(たとえば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を設定します。したがって、バッチスクリプトに2つのコマンドがあり、最初のコマンドが失敗した場合、2番目のコマンドが成功した後でも、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