我从我的shell脚本启动了一个后台进程,我想在脚本完成时杀死这个进程。

如何从我的shell脚本得到这个过程的PID ?就我所能看到的变量$!包含当前脚本的PID,而不是后台进程。


当前回答

pgrep可以获取父进程的所有子pid。如前所述,$$是当前脚本的PID。所以,如果你想要一个脚本自己清理,这应该可以做到:

trap 'kill $( pgrep -P $$ | tr "\n" " " )' SIGINT SIGTERM EXIT

其他回答

你也可以使用pstree:

pstree -p user

这通常为“user”提供所有进程的文本表示,而-p选项提供进程id。据我所知,这并不取决于进程是否属于当前shell。它还显示了分叉。

pgrep可以获取父进程的所有子pid。如前所述,$$是当前脚本的PID。所以,如果你想要一个脚本自己清理,这应该可以做到:

trap 'kill $( pgrep -P $$ | tr "\n" " " )' SIGINT SIGTERM EXIT

还有一种更简单的方法可以杀死bash脚本的所有子进程:

pkill -P $$

- p标志与pkill和pgrep的工作方式相同——它获取子进程,只有使用pkill时子进程被杀死,使用pgrep时子pid被打印到标准输出。

你需要保存后台进程启动时的PID:

foo &
FOO_PID=$!
# do other stuff
kill $FOO_PID

您不能使用作业控制,因为作业控制是一个交互特性,并且与控制终端绑定。脚本不一定要附加终端,因此作业控制不一定可用。

如果你在开始工作后不能直接获得PID,你也可以尝试这个,稍后获得PID:

foo &

# do some stuff and then

pid=$(ps -aux | grep foo | tr -s ' ' | cut -d\  -f2)
kill $pid

解释:

Ps获取所有进程的信息include . command Grep正在为您的命令进行筛选 Tr是去掉重复的空格进行切割 cut是让你得到带有PID的列(在这里是2)