我从我的Shell脚本中启动了一个后台进程,当我的脚本结束时,我想要杀死这个进程。
如何从我的Shell脚本中获取该进程的PID?据我所知,变量$!
包含当前脚本的PID,而不是后台进程的PID。
我从我的Shell脚本中启动了一个后台进程,当我的脚本结束时,我想要杀死这个进程。
如何从我的Shell脚本中获取该进程的PID?据我所知,变量$!
包含当前脚本的PID,而不是后台进程的PID。
foo &
FOO_PID=$!
# do other stuff
kill $FOO_PID
您不能使用作业控制,因为它是交互式功能,与控制终端相关联。脚本可能根本没有连接终端,因此作业控制可能不可用。
foo
和$!
之间会不会有其他进程启动并获取该进程的PID而不是foo
的PID呢? - WiSaGaNfoo
恰好是多个管道命令(例如 tail -f somefile.txt | grep sometext
),那么这会给你带来麻烦。在这种情况下,如果你要查找的是tail
命令的PID,那么你将从$!
中得到grep
命令的PID。在这种情况下,你需要使用jobs
或ps
等命令。 - John Rixgrep
的pid,但如果你杀死它,当tail
尝试写入管道时,会产生一个SIGPIPE信号。但一旦你尝试进入任何复杂的进程管理/控制,bash/shell就变得相当麻烦。 - camh/bin/sh -c 'echo $$>/tmp/my.pid && exec program args' &
- sysfault 于2010年11月24日14:28。 - imz -- Ivan Zakharyaschev您可以使用jobs -l
命令来获取特定的作业。
^Z
[1]+ Stopped guard
my_mac:workspace r$ jobs -l
[1]+ 46841 Suspended: 18 guard
在这种情况下,46841是PID。help jobs
得知:jobs -p
是另一种选项,仅显示PID。$!
中保存 PID 是更具可移植性和更简单的方法,这就是当前被接受的答案所做的。 - tripleee$!
。 - Calimojobs
是正确的选择,但这篇帖子是作为回答一个关于如何从脚本中以编程方式实现此操作的问题。 - tripleee$$
是当前脚本的进程号$!
是上一个后台进程的进程号这里是一个 bash 会话的示例记录(%1
表示从 jobs
看到的后台进程的序号):
$ echo $$
3748
$ sleep 100 &
[1] 192
$ echo $!
192
$ kill %1
[1]+ Terminated sleep 100
echo%1
不能返回后台进程,而echo $!
可以。 - Timo$$
并不总是当前的 PID。例如,如果您在 bash
中定义一个新函数并在后台运行该函数,则该函数内部的 $$
包含启动该函数的进程的 PID。如果您需要任何给定代码实际运行的进程的 PID,则必须使用 $BASHPID
。 - Mikko Rantalainenpkill -P $$
-P
标志在 pkill
和 pgrep
中的工作方式相同 - 它获取子进程,只是使用 pkill
杀死子进程,而使用 pgrep
将子 PID 打印到 stdout。
bash -c 'bash -c "sleep 300 &"' &
后,执行 pgrep -P $$
将什么也不会显示,因为 sleep 不会成为你的 shell 的直接子进程。 - petre$$
指的是当前 shell。 - Timobash -c 'bash -c "sleep 300 &"' & ; pgrep -P $$
命令后,标准输出会显示 [1] <pid>
。至少它显示了一些东西,但这可能不是 pgrep
的输出。 - Timobash -c 'bash -c "sleep 10 & wait $!"' & sleep 0.1; pstree -p $$
- Pauli Nieminen这是我所做的。请查看,希望能对您有所帮助。
#!/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
pgrep
可以获取父进程的所有子PID。如前所述,$$
是当前脚本的PID。因此,如果您想要一个在执行完后自动清理的脚本,可以使用以下命令:
trap 'kill $( pgrep -P $$ | tr "\n" " " )' SIGINT SIGTERM EXIT
trap 'pkill -P $$' SIGING SIGTERM EXIT
看起来更简单,但我没有测试过。 - petreSIG
前缀。虽然 POSIX 允许这样做,但只是一种扩展,实现可能会支持:http://pubs.opengroup.org/onlinepubs/007904975/utilities/trap.html 例如,破折号 shell 就不支持。 - joschpgrep
,那么你也可能拥有 pkill
,这个更短的版本更为简洁。 - dragon788$$
将不会扩展。 - Patpstree -p user
这通常会给出所有“用户”进程的文本表示,-p选项则给出了进程ID。据我所知,它不依赖于当前Shell拥有进程的情况。它还显示分支。
如果您在启动作业后无法直接获取PID,则可以尝试以下方法稍后获取PID:
foo &
# do some stuff and then
pid=$(ps -aux | grep foo | tr -s ' ' | cut -d\ -f2)
kill $pid
解释:
ps
获取所有进程的信息,包括命令grep
过滤您的命令tr
去除重复空格以供 cut 使用cut
获取 PID 列(在此情况下为第二列)
bash
中,$$
可能是父进程ID:testfun() { echo "\$\$=$$ \$BASHPID=$BASHPID"; }; echo "my pid is $$"; testfun & wait
- Mikko Rantalainen