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

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

有帮助吗?

解决方案

您需要保存当时的后台进程的PID你开始吧:

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

可以不使用作业控制,因为这是一个交互式的特征和绑在控制终端。脚本不一定会附着在所有这样的作业控制终端不一定是可用的。

其他提示

可以使用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。

  • $$ 是当前脚本的pid
  • $! 是最后一个后台进程的pid

这是 bash 会话的示例记录(%1 指的是后台进程的序号,如下所示 jobs):

$ echo $$
3748

$ sleep 100 &
[1] 192

$ echo $!
192

$ kill %1

[1]+  Terminated              sleep 100

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

pkill -P $$

-P标志以同样的方式与pkillpgrep - 它得到的子进程,只有pkill子进程被杀掉,并与pgrep孩子的PID被打印到stdout

这是我做了什么。检查出来,希望能有所帮助。

#!/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

您可能还可以使用pstree:

pstree -p user

这典型地给出了“用户”和-p选项的所有进程的文本表示给出了进程ID。它不依赖,据我了解,在具有进程当前shell所拥有。它也显示了叉。

pgrep可以让你所有的父进程的孩子的PID。如前所述$$是当前脚本PID。所以,如果你想,经过自身清理脚本,这应该做的伎俩:

trap 'kill $( pgrep -P $$ | tr "\n" " " )' SIGINT SIGTERM EXIT
许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top