쉘 스크립트에서 백그라운드 프로세스를 시작하고 스크립트가 끝나면이 프로세스를 종료하고 싶습니다.
쉘 스크립트에서이 프로세스의 PID를 얻는 방법은 무엇입니까? 내가 볼 수있는 한 변수 $!
에는 백그라운드 프로세스가 아닌 현재 스크립트의 PID가 포함되어 있습니다.
쉘 스크립트에서 백그라운드 프로세스를 시작하고 스크립트가 끝나면이 프로세스를 종료하고 싶습니다.
쉘 스크립트에서이 프로세스의 PID를 얻는 방법은 무엇입니까? 내가 볼 수있는 한 변수 $!
에는 백그라운드 프로세스가 아닌 현재 스크립트의 PID가 포함되어 있습니다.
답변:
시작할 때 백그라운드 프로세스의 PID를 저장해야합니다.
foo &
FOO_PID=$!
# do other stuff
kill $FOO_PID
작업 제어는 대화식 기능이며 제어 터미널에 연결되어 있으므로 작업 제어를 사용할 수 없습니다. 스크립트에는 터미널이 전혀 연결되어 있지 않아도되므로 작업 제어가 반드시 필요한 것은 아닙니다.
foo
와 사이에 무언가가 시작될 수 있습니까? $!
그리고 우리는 그 대신에 pid를 얻습니다 foo
.
foo
여러 개의 배관 명령 (예 :)이 발생하는 경우 호스를 사용하십시오 tail -f somefile.txt | grep sometext
. 그러한 경우, $!
찾고자하는 경우 tail 명령 대신 grep 명령의 PID를 얻을 수 있습니다. 이 경우 jobs
또는 등 을 사용해야 ps
합니다.
grep
만 그것을 죽이면 tail은 파이프에 쓰려고 할 때 SIGPIPE를 얻습니다. 그러나 까다로운 프로세스 관리 / 제어를 시도하자마자 bash / shell은 상당히 고통스러워집니다.
/bin/sh -c 'echo $$>/tmp/my.pid && exec program args' &
sysfault 11월 24일 '10 14시 28분에서 -
이 jobs -l
명령을 사용하여 특정 작업을 수행 할 수 있습니다
^Z
[1]+ Stopped guard
my_mac:workspace r$ jobs -l
[1]+ 46841 Suspended: 18 guard
이 경우 PID는 46841입니다.
보낸 사람 help jobs
:
-l 프로세스 그룹 ID 및 작업의 작업 디렉토리를보고하십시오.
jobs -p
PID 만 보여주는 또 다른 옵션입니다.
$!
를 시작한 직후 부터 저장하면 대부분의 상황에서 이식성이 더 쉽고 간단합니다. 이것이 현재 받아 들여지는 대답입니다.
jobs -p
jobs -l
Lubuntu 16.4
bash 스크립트의 모든 자식 프로세스를 종료하는 더 간단한 방법 :
pkill -P $$
-P
플래그와 동일한 방식으로 작동 pkill
하고 pgrep
만 함께, 자식 프로세스를 가져옵니다 - pkill
와 자식 프로세스가 사망하고 얻을 pgrep
자식의 PID가 표준 출력에 인쇄됩니다.
bash -c 'bash -c "sleep 300 &"' &
달리기 후에 pgrep -P $$
는 아무것도 보이지 않습니다.
$$
현재 쉘을 의미합니다.
bash -c 'bash -c "sleep 300 &"' & ; pgrep -P $$
, 나는 stdout [1] <pid>. So at least it shows something, but this is probably not the output of
pgrep에 도착
bash -c 'bash -c "sleep 10 & wait $!"' & sleep 0.1; pstree -p $$
이것이 내가 한 일입니다. 그것을 확인하고 도움이되기를 바랍니다.
#!/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
더 단순 해 보이지만 테스트하지는 않았습니다.
SIG
접두사를 사용하지 마십시오 . POSIX에서 허용하지만 구현에서 지원할 수 있는 확장으로 다음과 같습니다. pubs.opengroup.org/onlinepubs/007904975/utilities/trap.html 예를 들어 대시 셸 은 지원 하지 않습니다.