如何获得后台进程的PID?

Vol*_*lyy 349 linux shell pid background-process

我从我的shell脚本开始后台进程,我想在脚本完成时终止这个进程.

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

cam*_*amh 534

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

foo &
FOO_PID=$!
# do other stuff
kill $FOO_PID
Run Code Online (Sandbox Code Playgroud)

您无法使用作业控制,因为这是一个交互式功能并且与控制终端绑定.脚本根本不需要连接终端,因此不一定可以使用作业控制.

  • @WiSaGaN:没有.这些线之间没有任何东西.系统上的任何其他活动都不会影响这一点.$!将扩展到shell_中最后一个后台进程_的PID. (46认同)
  • 因为$!返回最后一个后台进程的pid.是否有可能在`foo`和`$!`之间开始,我们得到的东西是pid而不是`foo`? (18认同)
  • ...如果`foo`碰巧是多个管道命令(例如``tail -f somefile.txt | grep sometext`),它会给你带来麻烦.在这种情况下,如果你正在寻找的话,你将从`$!`而不是tail命令获得grep命令的PID.在这种情况下,您需要使用`jobs`或`ps`等. (8认同)
  • 建议的另一个有价值的解决方案(对答案的评论)[如何获得刚开始的过程的pid](http://serverfault.com/a/205563/68972):哦,和"oneliner":`/ bin/sh -c'echo $$>/tmp/my.pid && exec program args'&` - sysfault 2010年11月24日14:28 (5认同)
  • @JohnRix:不一定。您将获得 `grep` 的 pid,但是如果您取消它,当它尝试写入管道时,tail 将获得一个 SIGPIPE。但是一旦您尝试进入任何棘手的进程管理/控制,bash/shell 就会变得非常痛苦。 (2认同)

jld*_*ont 135

您可以使用该jobs -l命令访问特定的jobL

^Z
[1]+  Stopped                 guard

my_mac:workspace r$ jobs -l
[1]+ 46841 Suspended: 18           guard
Run Code Online (Sandbox Code Playgroud)

在这种情况下,46841是PID.

来自help jobs:

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

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

  • @Phil仅列出pids:jobs -p.列出某个工作的pid:jobs -p%3.无需处理输出. (6认同)
  • 要在 shell 脚本中使用它,您必须处理输出。 (2认同)
  • 在启动后立即从“$!”中保存 PID,在大多数情况下更便携、更直接。这就是当前接受的答案的作用。 (2认同)
  • @Calimo 我同意“jobs”是交互式会话中的方法,但这是作为对一个问题的答案而发布的,特别是关于如何从脚本以编程方式执行此操作的问题。 (2认同)

cat*_*alk 45

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

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

$ echo $$
3748

$ sleep 100 &
[1] 192

$ echo $!
192

$ kill %1

[1]+  Terminated              sleep 100
Run Code Online (Sandbox Code Playgroud)


Ale*_*sky 25

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

pkill -P $$
Run Code Online (Sandbox Code Playgroud)

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

  • @AlexeyPolonsky:应该是:杀死 shell 的所有子进程,而不是脚本。因为`$$`指的是当前的shell。 (2认同)

小智 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
Run Code Online (Sandbox Code Playgroud)

然后只需将其运行为:./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
Run Code Online (Sandbox Code Playgroud)