检查POSIX中退出的子进程?

时间:2011-05-08 02:00:16

标签: c++ posix child-process waitpid

我正在尝试编写一个程序来检查已退出的子进程,并在它们退出时重新启动它们。它需要在进程退出时重新启动进程,而不必等待任何其他进程退出。我有点失落。这是我到目前为止所做的代码。它没有完成或正确,真的。但也许有人可以指出我正确的方向?

for(int ifile = 1; ifile < 4; ifile++){

    child_pid[ifile - 1] = vfork();

    if(child_pid[ifile - 1] == -1){
        cerr << "fork error on " << argv[ifile] << endl;
    }
    else if(child_pid[ifile - 1] == 0){
        execl(argv[ifile], argv[ifile], NULL);
    }
}

for(int index = 0; index < 3; index++){
    do{
        wait_pid = waitpid(child_pid[index], &status, WUNTRACED);
        if(WIFEXITED(status)){
            count++;
            child_pid[index] = vfork();
            if(child_pid[index] == -1){
                cerr << "rescheduled process error" << endl;
                return -1;
            }
            else if(child_pid[index] == 0){
                cout << "Rescheduling " << argv[index + 1] << endl;
                execl(argv[index + 1], argv[index + 1], NULL);
            }
        }
    }while(count != 4);
}

2 个答案:

答案 0 :(得分:3)

你的方法存在的问题是你要阻止waitpid,直到第一个[0]个孩子死亡,其他任何一个孩子可能在此期间死亡。

你可以:

(1)使用wait()而不是waitpid()。这将处理任何一个孩子,但它会阻止。

(2)在循环时使用waitpid()和WNOHANG标志对死去的孩子进行轮询。

(3)设置信号处理程序并捕获SIGCHILD信号。将waitpid / WNOHANG循环放在你的处理程序中,但是在它之外重新开始工作。

E.G。 Handler做了类似的事情:

   while ((child_pid = waitpid(-1, &status, WNOHANG)) > 0) 
   {
        //set a flag or whatever; not wise to fork/exec in handler
   }

答案 1 :(得分:2)

我不确定您为何需要WUNTRACED选项。如果它说“停止”,我很确定它意味着与暂停相同,而不是退出。

两种可能的方法:

而不是循环,只需:

pid_t pid = wait();

它将阻止,直到一个过程完成。

或者,将waitpid()的第三个参数从WUNTRACED更改为WNOHANG。这将让它在不阻塞的情况下测试每个进程。在这种情况下,您可能希望在每次传递结束时通过循环添加一个小睡眠。