如何向子进程发送信号?

时间:2019-01-02 20:50:50

标签: go signals

我无法从父进程发送信号并在子进程中接收信号。

这是子进程的代码。收到SIGINT时退出。

// child.go
func main() {
    stop := make(chan os.Signal, 1)
    signal.Notify(stop, os.Interrupt)
    fmt.Println("started")

    <-stop
    fmt.Println("stopped")
}

这是父进程。它开始child.go,发送SIGINT,然后等待它退出。

// main.go
func main() {
    // Start child process
    cmd := exec.Command("go", "run", "child.go")
    cmd.Stdout = os.Stdout
    cmd.Stderr = os.Stderr
    err := cmd.Start()
    if err != nil {
        _, _ = fmt.Fprintf(os.Stderr, "Start: " + err.Error())
        return
    }

    // Wait, then send signal
    time.Sleep(time.Millisecond * 500)
    err = cmd.Process.Signal(os.Interrupt)
    if err != nil {
        _, _ = fmt.Fprintf(os.Stderr, "Signal: " + err.Error())
        return
    }

    // Wait for child process to finish
    err = cmd.Wait()
    if err != nil {
        _, _ = fmt.Fprintf(os.Stderr, "Wait: " + err.Error())
    }
    return
}

此代码应打印started\nstopped以表明其按预期工作,但仅打印started并挂在cmd.Wait()上,这意味着子进程未接收到信号。

当我运行go run child.go时,它运行良好,因此我认为该文件没有问题。我了解func (*Process) Signal在Windows上不起作用;我正在使用Linux。

如何修复代码,以便子进程获取父进程发送的信号?

1 个答案:

答案 0 :(得分:2)

如@JimB在评论部分中所述,go run是您的问题。

go run child.go将编译child并将其作为自己的进程执行。如果您在ps之后运行go run child.go,则会看到两个进程正在运行。

您正在监视和发出信号的进程是go可执行文件,而不是child

用已编译的二进制文件exec.Command("go", "run", "child.go")代替exec.Command("child"),它应该可以工作。