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