1. 程式人生 > >linux命令(26):Bash Shell 獲取程序 PID

linux命令(26):Bash Shell 獲取程序 PID

Linux 的互動式 Shell 與 Shell 指令碼存在一定的差異,主要是由於後者存在一個獨立的執行程序,因此在獲取程序 pid 上二者也有所區別。

互動式 Bash Shell 獲取程序 pid

在已知程序名(name)的前提下,互動式 Shell 獲取程序 pid 有很多種方法,典型的通過 grep 獲取 pid 的方法為(這裡新增 -v grep是為了避免匹配到 grep 程序):

ps -ef | grep "name" | grep -v grep | awk '{print $2}'

或者不使用 grep(這裡名稱首字母加[]的目的是為了避免匹配到 awk 自身的程序):

ps -ef | awk '/[n]ame/{print $2}'

如果只使用 x 引數的話則 pid 應該位於第一位:

ps x | awk '/[n]ame/{print $1}'

最簡單的方法是使用 pgrep

pgrep -f name

如果需要查詢到 pid 之後 kill 掉該程序,還可以使用 pkill

pkill -f name

如果是可執行程式的話,可以直接使用 pidof

pidof name

Bash Shell 指令碼獲取程序 pid

根據程序名獲取程序 pid

在使用 Shell 指令碼獲取程序 pid 時,如果直接使用上述命令,會出現多個 pid 結果,例如

1
2
3
4
5
#! /bin/bash
# process-monitor.sh
process=$1
pid=$(ps x | grep $process | grep -v grep | awk '{print $1}')
echo $pid

執行 process-monitor.sh 會出現多個結果:

$> sh process-monitor.sh
3036  3098  3099

進一步排查可以發現,多出來的幾個程序實際上是子 Shell 的(臨時)程序:

root      3036  2905  0 09:03 pts/1    00:00:45 /usr/java/jdk1.7.0_71/bin/java ...name
root      4522  2905  0 16:12 pts/1    00:00:00 sh process-monitor.sh name
root      4523  4522  0 16:12 pts/1    00:00:00 sh process-monitor.sh name

其中 3036 是需要查詢的程序pid,而 4522、4523 就是子 Shell 的 pid。 為了避免這種情況,需要進一步明確查詢條件,考慮到所要查詢的是 Java 程式,就可以通過 Java 的關鍵字進行匹配:

1
2
3
4
5
#! /bin/bash
# process-monitor.sh
process=$1
pid=$(ps -ef | grep $process | grep '/bin/java' | grep -v grep | awk '{print $2}')
echo $pid

獲取 Shell 指令碼自身程序 pid

這裡涉及兩個指令: 1. $$ :當前 Shell 程序的 pid 2. $! :上一個後臺程序的 pid 可以使用這兩個指令來獲取相應的程序 pid。例如,如果需要獲取某個正在執行的程序的 pid(並寫入指定的檔案):

myCommand && pid=$!
myCommand & echo $! >/path/to/pid.file

注意,在指令碼中執行 $! 只會顯示子 Shell 的後臺程序 pid,如果子 Shell 先前沒有啟動後臺程序,則沒有輸出。

檢視指定程序是否存在

在獲取到 pid 之後,還可以根據 pid 檢視對應的程序是否存在(執行),這個方法也可以用於 kill 指定的程序。

if ps -p $PID > /dev/null
then
   echo "$PID is running"
   # Do something knowing the pid exists, i.e. the process with $PID is running
fi