【问题标题】:Don't want to remove terminated child process immediately, need to become zombie不想立即删除终止的子进程,需要变成僵尸
【发布时间】:2014-05-28 16:50:47
【问题描述】:

我从SE QUE得到以下信息 将 SIGCHLD 的配置显式设置为 SIG_IGN 会导致任何随后终止的子进程立即从系统中删除,而不是被转换为僵尸。

据我所知,要读取子状态,进程表中需要有子 pid。所以有必要在进程表中有僵尸子进程来读取子状态。

所以我想编写信号处理程序,它将删除“将 SIGCHLD 的处置设置为 SIG_IGN”

我使用下面的代码(在 fork 之前)避免在终止后立即删除子进程:但我仍然无法获得子进程并且 waitpid 返回 -1 和 ECHILD。

#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

static siginfo_t sig_info;
static volatile sig_atomic_t sig_num;
static void *sig_ctxt;

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\n", sig_info.si_signo, (int)sig_info.si_pid);
        sig_num = 0;
    }
}

请提出一些想法,我被这个阻止了。

【问题讨论】:

  • 如果您阅读关于 wait(2) 的注释,您会发现显式设置 SIG_IGN 会导致立即收获孩子。但是,默认操作(也是“忽略”)不会执行此操作。因此,要获得僵尸,您所要做的就是为 CIGCHLD 设置信号操作,或者,AFAIK,将其设置为 SIG_DFL。
  • 如何不为 SIGCHLD 或 AFAIK 设置信号操作?将其设置为 SIG_DFL?您能否提供示例实现链接?
  • 只需不要为 SIGCHLD 设置处理程序!在您显示的代码中,切勿调用 set_handler(SIGCHLD)。我可以从个人经验告诉你,如果你不使用 SIGCHLD,你会得到僵尸。

标签: c linux signals waitpid sigchld


【解决方案1】:

不要捕获信号,只需在 main 中使用此代码:

signal(SIGCHLD,SIG_IGN);

不要使用处理程序,而是使用 SIG_IGN(如上所示)。这忽略了信号,进程 pid 将保留在进程表中。然后,父进程可以使用 waitpid() 或 wait() 声明此僵尸进程的状态。

【讨论】:

  • 这是错误的。使用调用信号(SIGCHLD,SIG_IGN)显式忽略 SIGCHLD 将使终止的进程自动从进程表中删除(至少在内核 2.6 上)。请参阅 man waitpid 的注释部分。如果您希望它们留在表中,只需保留默认的信号处理程序即可。
猜你喜欢
  • 1970-01-01
  • 2016-07-03
  • 1970-01-01
  • 2013-05-01
  • 2017-10-16
  • 2013-06-05
  • 1970-01-01
  • 2021-07-26
  • 1970-01-01
相关资源
最近更新 更多