gpt4 book ai didi

c - 使用 C 以编程方式检查 Linux 中的僵尸子进程

转载 作者:太空狗 更新时间:2023-10-29 17:26:20 26 4
gpt4 key购买 nike

我在 RedHat Linux 中编写了一个简单的 C 程序,它在调用 execv 后使用 waitpid 等待子进程。

int main( int argc, char * argv[] )
{
int pid;
int status = 0;
int wait_ret;

const char * process_path = argv[1];

if ( argc < 2 )
{
exit( EXIT_FAILURE );
}

pid = fork(); //spawn child process

if ( 0 == pid ) //child
{
int ret = execv( process_path, &argv[1] );

if ( ret )
{
printf( "execv failed: %s\n", strerror( errno ) );
}

exit( EXIT_SUCCESS );
}

//wait for the child to terminate
wait_ret = waitpid( pid, &status, WUNTRACED );

if ( -1 == wait_ret )
{
printf( "ERROR: Failed to wait for process termination\n" );
exit( EXIT_FAILURE );
}

// ... handlers for child exit status ...

return 0;
}

我将其用作我正在运行的某些进程的简单看门狗。

我的问题是,特别是一个进程在退出时没有被 waitpid 收割,而是在 waitpid 挂起时永远处于僵尸状态。我不确定为什么 waitpid 一旦变成僵尸(可能是泄漏的文件描述符或其他东西)就无法获取此进程。

我可以使用 WNOHANG 标志并轮询 child 的 stat proc 文件来检查僵尸状态,但我更喜欢更优雅的解决方案。也许我可以使用某些功能来获取 Zombie 状态而无需轮询此文件?

有谁知道 waitpid 的替代方法,它会在进程变成僵尸时返回?

附加信息:

在其中一个线程中调用 exit( EXIT_FAILURE); 正在关闭子进程。

cat /proc/<CHILD_PID>/stat(退出前):

1037(MY_PROGRAM)S 1035 58 58 0 -1 4194560 1309 0 22 0 445 1749 0 20 0 20 0 13 0 13 0 4399 2234776 1136 4294967295 33367162880 17 0 0 0 26 0 0 3338489412 3338507560 3338600448

cat /proc/<CHILD_PID>/stat(退出后):

1037 (我的程序) Z 1035 58 58 0 -1 4227340 1316 0 22 0 464 1834 0 0 20 0 2 0 4399 0 0 4294967295 0 0 0 0 0 0 0 4 31850 4294967217 0 0 6 0 0 2 0 0 0

请注意,在这种情况下,子 PID 为 1037,父 PID 为 1035。

最佳答案

任何终止的进程都会变成僵尸,直到它被 wait 调用收集起来。这里的等待似乎并不是在所有情况下都会发生。

从给出的代码中我无法弄清楚为什么等待没有发生并且进程仍然是僵尸。 (反正不是没有运行它)

但不是仅等待特定的 pid,您可以通过使用 -1 作为 waitpid 的第一个参数来等待 任何 child .不要使用 WNOHANG,因为它需要繁忙的轮询(不要那样做)。

您可能还想删除 WUNTRACED 除非您有特定的理由将其包含在内。但是放弃它并看看它有什么不同没有坏处。

关于c - 使用 C 以编程方式检查 Linux 中的僵尸子进程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50541461/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com