假设我运行ps axf,我可以看到我的命令的进程树如下所示:
800 ? Ss 0:00 /usr/sbin/sshd 10186 ? Ss 0:00 \_ sshd: yukondude [priv] 10251 ? S 0:00 \_ sshd: yukondude@pts/0 10252 pts/0 Ss 0:00 \_ -bash 10778 pts/0 S 0:00 \_ su - 10785 pts/0 S 0:00 \_ -su 11945 pts/0 R+ 0:00 \_ ps axf
我知道我可以检查$$的当前shell的PID(10785)或$ PPID为父PID(10778)。
但是,我只想要顶级的父级PID,在这个例子中它是800(SSH守护进程)。有什么办法吗?
从this SO answer学到,我可以递归地检查/ proc / PID / stat文件中的第4个条目来查找每个进程的父PID:
# cut -f4 -d' ' /proc/10785/stat 10778 # cut -f4 -d' ' /proc/10778/stat 10252 # cut -f4 -d' ' /proc/10252/stat 10251 # cut -f4 -d' ' /proc/10251/stat 10186 # cut -f4 -d' ' /proc/10186/stat 800 # cut -f4 -d' ' /proc/800/stat 1
(顶级父级PID将是在我到达init的PID之前的一个PID,即1.)
在我写一个循环之前(我甚至不知道如果你可以在bash中使用递归)来做到这一点,是否有一个更简单的方法,我失踪了?也许只是/ proc下的一个文件的另一个参数?通过这些文件的grep没有显示任何明显的东西。
编辑:当然,所有Linux进程的顶级进程是/ sbin / init,PID为1.我想要的是父对象之前的PID:倒数第二个父进程。
没有一个更好的解决方案,这是一个简单的(递归)脚本,以获得您给出的任何进程号的顶级父PID(或当前shell如果省略PID参数):
#!/bin/bash # Look up the top-level parent Process ID (PID) of the given PID,or the current # process if unspecified. function top_level_parent_pid { # Look up the parent of the given PID. pid=${1:-$$} stat=($(</proc/${pid}/stat)) ppid=${stat[3]} # /sbin/init always has a PID of 1,so if you reach that,the current PID is # the top-level parent. Otherwise,keep looking. if [[ ${ppid} -eq 1 ]] ; then echo ${pid} else top_level_parent_pid ${ppid} fi }
只需输入此脚本,并根据需要调用带或不带PID参数的top_level_parent_pid。
感谢@Dennis Williamson对于如何紧凑而有效地编写这个脚本的许多建议。