Как получить PID фонового процесса? - PullRequest
328 голосов
/ 15 декабря 2009

Я запускаю фоновый процесс из своего сценария оболочки и хочу завершить этот процесс, когда мой сценарий завершится.

Как получить PID этого процесса из моего сценария оболочки? Насколько я вижу, переменная $! содержит PID текущего скрипта, а не фоновый процесс.

Ответы [ 7 ]

497 голосов
/ 16 декабря 2009

Вам нужно сохранить PID фонового процесса во время его запуска:

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

Вы не можете использовать управление заданиями, поскольку это интерактивная функция, связанная с управляющим терминалом. К сценарию не обязательно будет присоединен терминал вообще, поэтому управление заданиями не обязательно будет доступно.

125 голосов
/ 15 декабря 2009

Вы можете использовать команду jobs -l, чтобы перейти к конкретному заданиюL

^Z
[1]+  Stopped                 guard

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

В этом случае 46841 - это PID.

С help jobs:

-l Сообщить идентификатор группы процессов и рабочий каталог заданий.

jobs -p - это еще одна опция, которая показывает только PID.

41 голосов
/ 15 декабря 2009
  • $$ - это pid текущего скрипта
  • $! pid последнего фонового процесса

Вот пример стенограммы из сеанса bash (%1 относится к порядковому номеру фонового процесса, как видно из jobs):

$ echo $$
3748

$ sleep 100 &
[1] 192

$ echo $!
192

$ kill %1

[1]+  Terminated              sleep 100
25 голосов
/ 28 апреля 2014

Еще более простой способ уничтожить весь дочерний процесс скрипта bash:

pkill -P $$

Флаг -P работает аналогично с pkill и pgrep - он получает дочерние процессы, только с pkill дочерние процессы уничтожаются и с pgrep дочерние PID печатаются в стандартный вывод. *

3 голосов
/ 21 февраля 2013

это то, что я сделал. Проверьте это, надеюсь, это поможет.

#!/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
2 голосов
/ 18 мая 2012

Вы также можете использовать pstree:

pstree -p user

Это обычно дает текстовое представление всех процессов для "пользователя", а опция -p дает идентификатор процесса. Насколько я понимаю, это не зависит от того, будут ли процессы принадлежать текущей оболочке. Это также показывает вилки.

1 голос
/ 02 мая 2013

pgrep может получить все дочерние PID родительского процесса. Как уже упоминалось ранее $$ - это текущий PID скриптов. Итак, если вам нужен скрипт, который очищается после себя, это должно сработать:

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