我正在尝试编写一个程序来检查已退出的子进程,并在它们退出时重新启动它们。它需要在进程退出时重新启动进程,而不必等待任何其他进程退出。我有点失落。这是我到目前为止所做的代码。它没有完成或正确,真的。但也许有人可以指出我正确的方向?
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);
}
答案 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
。这将让它在不阻塞的情况下测试每个进程。在这种情况下,您可能希望在每次传递结束时通过循环添加一个小睡眠。