454

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

如何从我的 shell 脚本中获取此进程的 PID?据我所见,变量$!包含当前脚本的 PID,而不是后台进程。

4

8 回答 8

695

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

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

您不能使用作业控制,因为这是一项交互功能并且与控制终端相关联。脚本不一定要连接终端,因此作业控制不一定可用。

于 2009-12-16T00:05:40.370 回答
174

您可以使用该jobs -l命令来获取特定的 jobL

^Z
[1]+  Stopped                 guard

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

在这种情况下,46841 是 PID。

来自help jobs

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

jobs -p是另一个仅显示 PID 的选项。

于 2009-12-15T16:27:22.453 回答
54
  • $$是当前脚本的 pid
  • $!是最后一个后台进程的pid

这是来自 bash 会话的示例记录(%1指从 中看到的后台进程的序号jobs):

$ echo $$
3748

$ sleep 100 &
[1] 192

$ echo $!
192

$ kill %1

[1]+  Terminated              sleep 100
于 2009-12-15T16:28:22.590 回答
29

杀死 bash 脚本的所有子进程的更简单方法:

pkill -P $$

-P标志与pkilland的工作方式相同pgrep- 它获取子进程,只有pkill子进程被杀死并且pgrep子 PID 被打印到标准输出。

于 2014-04-28T08:52:56.887 回答
5

这就是我所做的。检查一下,希望它可以帮助。

#!/bin/bash
#
# So something to show.
echo "UNO" >  UNO.txt
echo "DOS" >  DOS.txt
#
# Initialize Pid List
dPidLst=""
#
# Generate background processes
tail -f UNO.txt&
dPidLst="$dPidLst $!"
tail -f DOS.txt&
dPidLst="$dPidLst $!"
#
# Report process IDs
echo PID=$$
echo dPidLst=$dPidLst
#
# Show process on current shell
ps -f
#
# Start killing background processes from list
for dPid in $dPidLst
do
        echo killing $dPid. Process is still there.
        ps | grep $dPid
        kill $dPid
        ps | grep $dPid
        echo Just ran "'"ps"'" command, $dPid must not show again.
done

然后将其运行为:./bgkill.sh当然具有适当的权限

root@umsstd22 [P]:~# ./bgkill.sh
PID=23757
dPidLst= 23758 23759
UNO
DOS
UID        PID  PPID  C STIME TTY          TIME CMD
root      3937  3935  0 11:07 pts/5    00:00:00 -bash
root     23757  3937  0 11:55 pts/5    00:00:00 /bin/bash ./bgkill.sh
root     23758 23757  0 11:55 pts/5    00:00:00 tail -f UNO.txt
root     23759 23757  0 11:55 pts/5    00:00:00 tail -f DOS.txt
root     23760 23757  0 11:55 pts/5    00:00:00 ps -f
killing 23758. Process is still there.
23758 pts/5    00:00:00 tail
./bgkill.sh: line 24: 23758 Terminated              tail -f UNO.txt
Just ran 'ps' command, 23758 must not show again.
killing 23759. Process is still there.
23759 pts/5    00:00:00 tail
./bgkill.sh: line 24: 23759 Terminated              tail -f DOS.txt
Just ran 'ps' command, 23759 must not show again.
root@umsstd22 [P]:~# ps -f
UID        PID  PPID  C STIME TTY          TIME CMD
root      3937  3935  0 11:07 pts/5    00:00:00 -bash
root     24200  3937  0 11:56 pts/5    00:00:00 ps -f
于 2013-02-20T23:11:38.237 回答
3

您也许还可以使用 pstree:

pstree -p user

这通常为“用户”提供所有进程的文本表示,并且 -p 选项提供进程 ID。据我了解,它不依赖于让进程归当前外壳所有。它还显示了叉子。

于 2012-05-18T16:43:06.820 回答
2

pgrep可以获得父进程的所有子 PID。前面提到的$$是当前脚本的PID。因此,如果您想要一个自行清理的脚本,这应该可以解决问题:

trap 'kill $( pgrep -P $$ | tr "\n" " " )' SIGINT SIGTERM EXIT
于 2013-05-02T05:49:12.293 回答
0

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

于 2021-03-23T19:48:37.287 回答