導讀
Linux 的交互式 Shell 與 Shell 腳本存在必定的差別,主要是因爲後者存在一個獨立的運行進程,所以在獲取進程 pid 上兩者也有所區別。java
在已知進程名(name
)的前提下,交互式 Shell 獲取進程 pid 有不少種方法,典型的經過 grep 獲取 pid 的方法爲(這裏添加-v grep
是爲了不匹配到 grep 進程):bash
ps -ef | grep "name" | grep -v grep | awk '{print $2}'
spa
或者不使用 grep
(這裏名稱首字母加[]
的目的是爲了不匹配到 awk 自身的進程):code
ps -ef | awk '/[n]ame/{print $2}'
進程
若是隻使用 x 參數的話則 pid 應該位於第一位:it
ps x | awk '/[n]ame/{print $1}'
table
最簡單的方法是使用 pgrep
:awk
pgrep -f name
後臺
若是須要查找到 pid 以後 kill 掉該進程,還能夠使用 pkill
:jdk
pkill -f name
若是是可執行程序的話,能夠直接使用 pidof
pidof name
在使用 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,而 452二、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 |
這裏涉及兩個指令: 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