35

はいの場合、どのオペレーティング システム、シェルなどでしょうか?

次の Java プログラムを考えてみましょう (例として Java を使用しています。オペレーティング システムに関するこの質問にはどの言語でも適しています)。

public class ExitCode {
    public static void main(String args[]) {
        System.exit(Integer.parseInt(args[0]));
    }
}

Linux と bash で実行すると、常に 255 より小さい値が返されます (echo $?前に実行されたコマンドの終了コードを出力します)。

> java ExitCode 2; echo $?
2

> java ExitCode 128; echo $?
128

> java ExitCode 255; echo $?
255

> java ExitCode 256; echo $?
0

> java ExitCode 65536; echo $?
0

編集済み:以下の(これまでのところ)回答は、UNIXで何が起こるかを完全に説明しています。私はまだ他のOSについて疑問に思っています。

4

3 に答える 3

36

wait()またはを使用するwaitpid()

Unixおよび派生物では、wait()およびのようなPOSIX関数を使用することはできませんwaitpid()。返される終了ステータス情報は、2つの8ビットフィールドで構成されます。1つは終了ステータスを含み、もう1つは死因に関する情報を含みます(0はプログラム制御下での正常な終了を意味し、他の値はシグナルがそれを強制終了したことを示し、コアがダンプされました)。

で使用sigaction()するSA_SIGINFO

一生懸命働いて、sigaction()and<signal.h>Signal Actionsexit()のPOSIX仕様を読むと、子プロセスから渡された32ビット値を取得できることがわかります。ただし、それは完全に単純ではありません。

#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/wait.h>
#include <time.h>
#include <unistd.h>

static siginfo_t sig_info = { 0 };
static volatile sig_atomic_t sig_num = 0;
static void *sig_ctxt = 0;

static void catcher(int signum, siginfo_t *info, void *vp)
{
    sig_num = signum;
    sig_info = *info;
    sig_ctxt = vp;
}

static void set_handler(int signum)
{
    struct sigaction sa;
    sa.sa_flags = SA_SIGINFO;
    sa.sa_sigaction = catcher;
    sigemptyset(&sa.sa_mask);

    if (sigaction(signum, &sa, 0) != 0)
    {
        int errnum = errno;
        fprintf(stderr, "Failed to set signal handler (%d: %s)\n", errnum, strerror(errnum));
        exit(1);
    }
}

static void prt_interrupt(FILE *fp)
{
    if (sig_num != 0)
    {
        fprintf(fp, "Signal %d from PID %d (status 0x%.8X; UID %d)\n",
                sig_info.si_signo, (int)sig_info.si_pid, sig_info.si_status,
                (int)sig_info.si_uid);
        sig_num = 0;
    }
}

static void five_kids(void)
{
    const int base = 0xCC00FF40;
    for (int i = 0; i < 5; i++)
    {
        pid_t pid = fork();
        if (pid < 0)
            break;
        else if (pid == 0)
        {
            printf("PID %d - exiting with status %d (0x%.8X)\n",
                   (int)getpid(), base + i, base + i);
            exit(base + i);
        }
        else
        {
            int status = 0;
            pid_t corpse = wait(&status);
            if (corpse != -1)
                printf("Child: %d; Corpse: %d; Status = 0x%.4X - waited\n", pid, corpse, (status & 0xFFFF));
            struct timespec nap = { .tv_sec = 0, .tv_nsec = 1000000 }; // 1 millisecond
            nanosleep(&nap, 0);
            prt_interrupt(stdout);
            fflush(0);
        }
    }
}

int main(void)
{
    set_handler(SIGCHLD);
    five_kids();
}

sigexit73(からコンパイルされたプログラムsigexit73.c)を実行すると、次のような出力が生成されます。

$ sigexit73
PID 26599 - exiting with status -872349888 (0xCC00FF40)
Signal 20 from PID 26599 (status 0xCC00FF40; UID 501)
Child: 26600; Corpse: 26599; Status = 0x4000 - waited
PID 26600 - exiting with status -872349887 (0xCC00FF41)
Signal 20 from PID 26600 (status 0xCC00FF41; UID 501)
Child: 26601; Corpse: 26600; Status = 0x4100 - waited
PID 26601 - exiting with status -872349886 (0xCC00FF42)
Signal 20 from PID 26601 (status 0xCC00FF42; UID 501)
Child: 26602; Corpse: 26601; Status = 0x4200 - waited
PID 26602 - exiting with status -872349885 (0xCC00FF43)
Signal 20 from PID 26602 (status 0xCC00FF43; UID 501)
Child: 26603; Corpse: 26602; Status = 0x4300 - waited
PID 26603 - exiting with status -872349884 (0xCC00FF44)
Signal 20 from PID 26603 (status 0xCC00FF44; UID 501)
$

1ミリ秒の呼び出しをnanosleep()削除すると、出力は次のようになりがちです。

$ sigexit73
sigexit23
PID 26621 - exiting with status -872349888 (0xCC00FF40)
Signal 20 from PID 26621 (status 0xCC00FF40; UID 501)
Child: 26622; Corpse: 26621; Status = 0x4000 - waited
PID 26622 - exiting with status -872349887 (0xCC00FF41)
PID 26623 - exiting with status -872349886 (0xCC00FF42)
Signal 20 from PID 26622 (status 0xCC00FF41; UID 501)
Child: 26624; Corpse: 26623; Status = 0x4200 - waited
Signal 20 from PID 26623 (status 0xCC00FF42; UID 501)
Child: 26625; Corpse: 26622; Status = 0x4100 - waited
PID 26624 - exiting with status -872349885 (0xCC00FF43)
PID 26625 - exiting with status -872349884 (0xCC00FF44)
$

ここから始まる行は3行だけであり、 ;Signalで終わる行も3行しかないことに注意してください。waited一部のシグナルと終了ステータスが失われます。SIGCHLDこれは、親プロセスに設定されているシグナル間のタイミングの問題が原因である可能性があります。

ただし、重要な点はexit()、コードがステータスを追跡するためsigaction()SIGCHLD、を使用する場合、ステータスで4バイトのデータを送信できることです。SA_SIGINFO

念のため、テストは、GCC9.2.0とXCode11.3.1を使用して、macOSMojave10.14.6を実行しているMacBookProで実行されました。このコードは、GitHubのSOQsigexit73.c (Stack Overflow Questions)リポジトリでも、src/so-1843-7779サブディレクトリのファイルとして入手できます。

于 2008-10-07T18:00:27.787 に答える
14

最新の Windows、OS 自体、およびデフォルトのコンソール シェル ( CMD.EXE) では、少なくとも 32 ビット符号付き整数の範囲全体で終了コードを受け入れて表示します。上記の例を実行するCMD.EXEと、要求した終了コードが得られます。

> java ExitCode 2
> echo %errorlevel%
2

> java ExitCode 128
> echo %errorlevel%
128

> java ExitCode 255
> echo %errorlevel%
255

> java ExitCode 256
> echo %errorlevel%
256

> java ExitCode 65536
> echo %errorlevel%
65536

Windows には実際には Unix シグナルの概念がありません。また、終了コードをハイジャックして追加情報を追加しようとすることもありません。そのため、シェル (または最終的に終了コードを読み取るプログラム) がそれを行わない限り、返された終了コードを取得する必要があります。さいわい、Microsoft の C ランタイムを使用するプログラム (MS Visual C++ でコンパイルされたすべてのプログラムを含む) は、終了プロセスからの終了コードをそのまま保持します。

于 2008-11-30T04:03:51.963 に答える
1

Windows にはさらに多くの終了コードがあり、14,000 を超えます。(自分の画面でそれらのいくつかをよく見たと思います)。

の登場:

于 2008-10-15T16:01:21.973 に答える