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

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


当前回答

可以使用jobs -l命令访问指定的jobL

^Z
[1]+  Stopped                 guard

my_mac:workspace r$ jobs -l
[1]+ 46841 Suspended: 18           guard

在本例中,PID为46841。

来自帮助工作:

-l报告任务的进程组ID和工作目录。

jobs -p是另一个选项,它只显示pid。

其他回答

可以使用jobs -l命令访问指定的jobL

^Z
[1]+  Stopped                 guard

my_mac:workspace r$ jobs -l
[1]+ 46841 Suspended: 18           guard

在本例中,PID为46841。

来自帮助工作:

-l报告任务的进程组ID和工作目录。

jobs -p是另一个选项,它只显示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)

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

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

$$是当前脚本的pid 美元!是上次后台进程的pid吗

下面是bash会话的示例记录(%1指从job中看到的后台进程的序数):

$ echo $$
3748

$ sleep 100 &
[1] 192

$ echo $!
192

$ kill %1

[1]+  Terminated              sleep 100

我在提供各种基础设施对象时多次遇到过这个问题。很多时候,您需要使用kubectl或临时端口转发临时代理。我发现超时命令是一个很好的解决方案,因为它允许我的脚本是自包含的,我可以确保进程将结束。我尝试设置小的超时,并重新运行脚本,如果我仍然需要它。