登录
首页 >  Golang >  Go问答

子进程为何在执行"sleep 10"时无法终止?

来源:stackoverflow

时间:2024-03-23 19:18:30 140浏览 收藏

当子进程执行 "sleep 10" 时,它进入了一种特殊的 "已失效" 状态,无法被终止。这是因为操作系统会将已完成但尚未被其上级进程收集的进程保留在该状态,以便上级进程可以通过调用 wait 系统调用来检测其终止。

问题内容

我需要测试一个进程是否终止,而我所拥有的只是它的 pid 号。为此,我测试伪文件“/proc/”是否存在。

为此函数编写测试时,我注意到该进程没有按预期终止。

对于测试,我运行“sleep 10”作为子进程,该子进程应运行 10 秒。启动此过程后,我轮询伪文件“/proc/”是否消失。该伪文件永远不会消失,并且不会检测到子进程的终止。

测试golang演示代码重现问题:https://play.golang.org/p/fb4cbxkijh3。

我检查了进程是否已创建,并且 pid 是否正确。检查进程时发现它变成了。它不会因此被删除。

问题如下:

  1. 为什么子进程没有终止?
  2. 如何更改代码以使其终止?
package main

import (
    "fmt"
    "log"
    "os"
    "os/exec"
    "strconv"
    "time"
)

func main() {
    fmt.Println("Hello, playground")

    cmd := exec.Command("sleep", "10")
    if err := cmd.Start(); err != nil {
        log.Fatal("unexpected error:", err)
    }
    pidStr := strconv.Itoa(cmd.Process.Pid)
    log.Println("sleep pid:", pidStr)

    for {
        if _, err := os.Stat("/proc/" + pidStr); os.IsNotExist(err) {
            log.Println("detect termination of /proc/" + pidStr)
            return
        }
        log.Println("pgm /proc/" + pidStr + " is running")
        time.Sleep(3 * time.Second)
    }
}

解决方案


在操作系统级别,在任何 POSIX 兼容操作系统(Unix、Linux、Darwin 等)中,已完成但尚未被其上级收集的进程处于“已失效”或“僵尸”状态。它仍然存在,但无法被杀死:它已经死了。它的存在正是为了使其上级(可以调用操作系统级别 wait 系统调用的进程)可以调用操作系统级别 wait 系统调用并看到该进程现在已死亡.

一旦其上级等待它,该进程就真正被删除:不再有僵尸进程占用该进程 ID。如果您有 /proc 文件系统,此时进程就会从 /proc 中消失。

在 Go 中,调用 cmd.Wait() 会调用操作系统级别的 wait 调用,因此这就是执行此操作的方法。如果您想收集此 cmd.Wait() 的结果,一个好方法是通过通道发送它。

(如果您想生成一个运行时间很长的进程而不等待它,您可以放弃它,这样您就不再是它的上级。有关正确执行此操作的详细信息充满了特定于操作系统的小玩意,例如丢弃控制 tty、设置会话、使用 procctlprctl 等。)

以上就是《子进程为何在执行"sleep 10"时无法终止?》的详细内容,更多关于的资料请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>