僵尸进程

一个进程使用fork创建子进程,如果子进程退出,而父进程没有调用wait或waitpid获取子进程的状态信息,那么子进程的进程描述符仍然保存在系统中,这种进程称为僵尸进程。

实现

僵尸进程是子进程结束,父进程不知道。
实现思路:父进程创建完子进程sleep 60s,子进程sleep 10s后退出。

package main

import (
	"fmt"
	"os"
	"time"
)

func main() {
	// 创建子进程
	attr := &os.ProcAttr{
		Dir:   "/tmp",
		Files: []*os.File{os.Stdin, os.Stdout, os.Stderr},
		Env:   os.Environ(),
	}
	cmd := "/bin/bash"
	args := []string{cmd, "-c", `sleep 10 && echo "child: exit"`}
	pro, err := os.StartProcess(cmd, args, attr)
	if err != nil {
		panic(err)
	}

	fmt.Printf("parent id: %d ,child id %d \n", os.Getpid(), pro.Pid)
	time.Sleep(60 * time.Second)
	fmt.Println("parent: wake up")
	
	// 为子进程收尸
    proState, err := pro.Wait()
    if err != nil {
        panic(err)
    }
    fmt.Println(proState.String())
}

在这里插入图片描述

孤儿进程

一个父进程退出,而它的一个或多个子进程还在运行,那么这些子进程将称为孤儿进程。孤儿进程将被init进程(进程号为1)所收养,并由init进程对他们完成状态收集工作。

实现

孤儿进程是在父进程死掉后还在执行自己任务的进程。
实现思路:生成一个子进程,子进程先sleep 30s 后输出退出信息。 父进程创建完子进程后sleep 10s后退出。在父进程sleep 10s期间查看子进程的父进程id, 父进程退出后,再查看子进程的id。

main.go:

package main

import (
    "fmt"
    "os"
    "time"
)

func main() {
    // 创建子进程
    attr := &os.ProcAttr{
        Dir:   "/tmp",
        Files: []*os.File{os.Stdin, os.Stdout, os.Stderr},
        Env:   os.Environ(),
    }
    cmd := "/bin/bash"
    args := []string{cmd, "-c", `sleep 30 && echo "child: exit"`}
    pro, err := os.StartProcess(cmd, args, attr)
    if err != nil {
        panic(err)
    }
    fmt.Printf("parent id: %d ,child id %d \n", os.Getpid(), pro.Pid)
    time.Sleep(10 * time.Second)
    fmt.Println("parent: exit")
}

在这里插入图片描述