8

我有一个想要监视另一个程序的 Bourne shell (/bin/sh) 脚本(用于可移植性)。它应该启动另一个程序,然后等待它退出。当第二个程序退出时,它会做一些最后的工作并自行退出。问题是脚本还需要响应信号(例如 USR2)并在这些信号出现时做一些工作。

我天真的实现是:

#! /bin/sh
echo $$
trap 'echo Respond to USR2' USR2
/bin/sleep 120 &
pid=$!
wait $pid
echo $pid exited with $?
echo Doing final cleanup

这行不通。如果我发送 shell SIGUSR2,陷阱会按预期触发,但等待也会结束,返回 140。 /bin/sleep 继续其愉快的方式。典型输出:

28849
Respond to USR2
28850 exited with 140
Doing final cleanup

这种行为在 dash 和 bash 之间是一致的,我可以方便地访问这两个 Bourne shell 衍生物。

我目前的工作是旋转循环等待子 PID 消失,用 kill 进行探测。自旋循环似乎很浪费,并且如果 PID 被快速重用,我的脚本可能会错误地等待错误进程的窗口扩大。

#! /bin/sh
echo $$
trap 'echo Respond to USR2' USR2
/bin/sleep 15 &
pid=$!
while /bin/kill -0 $pid 2> /dev/null; do
    echo waiting...
    sleep 2
done
echo Doing final cleanup

考虑到我的目标是同时等待另一个进程退出并能够响应信号,是否有更好的解决方案?

4

1 回答 1

3

你可以这样做:

while wait $pid; test $? -gt 128; do
    kill -0 $pid 2> /dev/null || break;
done

但请注意 sh 标准中的以下内容:

如果等待的退出状态大于 128,则应用程序无法知道等待的进程是以该值退出还是被信号杀死。由于大多数实用程序以较小的值退出,因此很少有任何歧义。即使在模棱两可的情况下,大多数应用程序只需要知道异步作业失败了;它是否检测到错误并失败或被杀死并且没有正常完成其工作都没有关系。

在这种情况下,歧义略有不同。您不知道等待是否被信号中断,或者孩子是否被信号终止。

于 2012-07-22T13:41:55.630 回答