2

我对 c 和编程非常陌生,需要一些帮助。在 linux(cygwin) 上的 c 中,我需要在退出时删除所有子进程。我已经查看了其他类似的问题,但无法使其正常工作。我试过了-

atexit(killzombies); //in parent process

void killzombies(void)
{
    printf("works");
    kill(0, SIGTERM);
    printf("works");
    if (waitpid(-1, SIGCHLD, WNOHANG) < 0)
         printf("works");
}

出于某种原因,“作品”甚至不会打印出来。我按 ctrl + c 退出。

我也试过-

prctl(PR_SET_PDEATHSIG, SIGHUP); //in child process
signal(SIGHUP, killMe);

void killMe()
{
    printf("works");
    exit(1);
}

但是因为我使用的是cygwin,当我使用cygwin时#include <sys/prctl.h>,cygwin说它找不到文件或目录,我不知道要为它安装什么包。另外,如果我的prctl()功能起作用,那会杀死所有的僵尸吗?

我的程序是一个客户端服务器,我的服务器 forks() 来处理每个客户端。我想当服务器关闭时不会留下任何剩余的僵尸。

4

3 回答 3

1

waitpid没有提供通常的参数,我很惊讶它没有崩溃。原型是:

pid_t waitpid(pid_t pid, int *status, int options); 

第二个参数应该是指向 的指针int您提供的是int. 还要注意,你应该waitpid每个孩子呼唤,你只是为一个孩子呼唤它。

atexit()仅当您正常退出时才会调用。如果您通过 CTRL+C 退出,那么您需要从 SIGINT 上的处理程序调用您的函数。

于 2012-09-14T15:37:41.503 回答
0

来自 Linux 文档atexit(3)

如果进程由于传递信号而异常终止,则不会调用使用atexit()(and )注册的函数。on_exit(3)

如果您想在应用程序收到 SIGINT 或 SIGTERM 时进行清理,您需要安装适当的信号处理程序并在那里完成您的工作。

于 2012-09-14T15:33:24.340 回答
0

您需要跟踪您的进程有多少孩子,然后多次调用 wait 。此外,正如其他人所说,如果进程被信号终止,则不会调用您的 atexit() 函数,因此您还需要从信号处理程序中调用 killzombies() 。你需要类似的东西:

int n_children = 0; // global

void handle_sig(int sig) {
    killzombies();
    exit(sig);
}

// your atexit()
void killzombies() {
    kill(0, SIGKILL);
    while (n_children > 0) {
        if (wait(NULL) != -1) {
            n_children--;
        }
    }
}
于 2012-09-14T16:09:11.443 回答