understanding-linux-processes

PPID

每个进程除了一定有PID还会有PPID,也就是父进程ID,通过PPID可以找到父进程的信息。

为什么进程都会有父进程ID呢?因为进程都是由父进程衍生出来的,后面会详细介绍几种衍生的方法。那么跟人类起源问题一样,父进程的父进程的父进程又是什么呢?实际上有一个PID为1的进程是由内核创建的init进程,其他子进程都是由它衍生出来,所以前面的描述并不准确,进程号为1的进程并没有PPID。

因为所有进程都来自于一个进程,所以Linux的进程模型也叫做进程树。

示例程序

要想获得进程的PPID,可以通过以下Getppid()这个函数来获得,print_ppid.go程序的代码如下。

package main

import (
  "fmt"
  "os"
  )

func main() {
    fmt.Println(os.Getppid())
}

运行结果

root@87096bf68cb2:/go/src# go run print_ppid.go
2892
root@87096bf68cb2:/go/src# go run print_ppid.go
2902

有趣的事情发生了,有没有发现每次运行的父进程ID都不一样,这不符合我们的预期啊,原来我们通过go run每次都会启动一个新的Go虚拟机来执行进程。

编译后运行

如果我们先生成二进制文件再执行结果会怎样呢?

root@87096bf68cb2:/go/src# ./print_ppid
1
root@87096bf68cb2:/go/src# ./print_ppid
1
root@87096bf68cb2:/go/src# ps aux |grep "1" |grep -v "ps" |grep -v "grep"
root         1  0.0  0.3  20228  3184 ?        Ss   07:25   0:00 /bin/bash

这次我们发现父进程ID都是一样的了,而且通过ps命令可以看到父进程就是bash,说明通过终端执行命令其实是从bash这个进程衍生出各种子进程。

为了执行这个程序要查找包依赖、编译、打包、链接(和go build做一样的东西)然后执行,这是全新的进程。

拿到PID和PPID后有什么用呢?马上揭晓。