导读
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