父进程传递给子进程无法处理的 SIGINT 信号,子进程突然终止
来源:stackoverflow
时间:2024-03-06 11:30:32 418浏览 收藏
今天golang学习网给大家带来了《父进程传递给子进程无法处理的 SIGINT 信号,子进程突然终止》,其中涉及到的知识点包括等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~
我正在尝试使用 stdpipes 在 golang 中管理一个应用程序(需要通过特定过程关闭,在本例中是拯救世界)。
这是我想要实现的目标的一个简单示例,但我有一个对我来说非常具体的问题,但对其他人来说也可能很有趣(也许您可以建议如何概括它)。
我还添加了一个名为 interruptlistener
的函数,该函数创建一个 goroutine 并在发送终止信号时管理程序的停止
脚本正常运行:
- 启动我的世界服务器
- 等待 40 秒,然后通过 stdin 发出“停止”命令
(在这种情况下,它按预期打印有关保存过程的所有日志)
测试用例(演示问题出在哪里):
- 启动我的世界服务器
- 在脚本发出停止命令之前,用户发送 ctrl+c
(在这种情况下,它应该完成打印有关保存过程的日志,然后退出,但它没有...似乎在收到终止信号后
scanner.scan()
返回 false,因此它只是退出)
您知道为什么会发生这种情况吗?我应该研究什么才能找到解决方案?
我真的迷路了,我已经花了 8 个多小时研究所有可能的代码组合......
package main import ( "bufio" "fmt" "io" "os" "os/exec" "os/signal" "strings" "sync" "syscall" "time" ) var cmd *exec.Cmd var wg sync.WaitGroup var stdOut io.ReadCloser var stdErr io.ReadCloser var stdIn io.WriteCloser func main() { interruptListener() cSplit := strings.Split("java -Xmx1024M -Xms1024M -jar server.jar nogui", " ") cmd = exec.Command(cSplit[0], cSplit[1:]...) cmd.Dir = "path/to/server/folder" stdOut, _ = cmd.StdoutPipe() stdErr, _ = cmd.StderrPipe() stdIn, _ = cmd.StdinPipe() wg.Add(2) go printer(stdOut) go printer(stdErr) err := cmd.Start() if err != nil { fmt.Println(err) } // test 1: wait 40 seconds for it to send the stop command and observe the output // test 2: in the 40 seconds (after the server has loaded press ctrl+c), there is no output time.Sleep(40 * time.Second) execute("stop") err = cmd.Wait() if err != nil { fmt.Println(err.Error()) } } func printer(stdP io.ReadCloser) { defer func() { wg.Done() fmt.Println("printer is out") }() var line string scanner := bufio.NewScanner(stdP) for scanner.Scan() { line = scanner.Text() fmt.Println(line) } fmt.Printf("scanner error: %v\n", scanner.Err()) } func execute(com string) { fmt.Println("sending", com, "to terminal") // needs to be added otherwise the virtual "enter" button is not pressed com += "\n" // write to cmd _, err := stdIn.Write([]byte(com)) if err != nil { fmt.Println(err.Error()) } } func interruptListener() { c := make(chan os.Signal, 1) signal.Notify(c, syscall.SIGHUP, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT) go func() { select { case <-c: execute("stop") wg.Wait() os.Exit(0) } }() }
解决方案
感谢在评论中回答的人,这是一个可以接受的解决方案(仍然不完美,如评论中所述)
只需在执行 cmd.start()
之前添加这几行:
// launch as new process group so that signals (ex: SIGINT) are not sent also the the child process cmd.SysProcAttr = &syscall.SysProcAttr{ CreationFlags: syscall.CREATE_NEW_PROCESS_GROUP, // windows // Setpgid: true, // linux }
这是您问题的更完整答案 - Killing a child process and all of its children in Go
今天关于《父进程传递给子进程无法处理的 SIGINT 信号,子进程突然终止》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习