一个进程使用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")
}