本篇主要是讲述一个go程序,如何在windows服务中优雅开启和关闭,废话不多说,开搞!!!
使用方式:go程序 + net服务启动
Windows go进程编译后,为一个.exe文件,直接执行即可运行
net start tamper_proof_refactored
启动后,发现windows服务根本不认识这个go二进制文件,完全起不来
注意:因为普通的程序无法处理Windows服务特有的消息,普通的Go程序也无法在服务模式运行。
现在知道一个问题了:普通的go程序启动服务后,服务会出现"无响应控制功能"
现在是想要实现什么?
go程序编译的二进制文件,可以被当作windows服务,稳定在后台执行,并可优雅开启和关闭
找到一个go的官方库,用于实现windows服务注册和启动的
golang.org/x/sys/windows/svc
具体实现逻辑如下
func (m *myService) Execute(args []string, r <-chan svc.ChangeRequest, changes chan<- svc.Status) (ssec bool, errno uint32) { const cmdsAccepted = svc.AcceptStop | svc.AcceptShutdown | svc.AcceptPauseAndContinue changes <- svc.Status{State: svc.StartPending} go Star() // 启动go程序的主程序【需要有一个主入口函数】 changes <- svc.Status{State: svc.Running, Accepts: cmdsAccepted} loop: for { c := <-r switch c.Cmd { case svc.Interrogate: changes <- c.CurrentStatus case svc.Stop, svc.Shutdown: Exit() // 停止你的程序 break loop case svc.Pause: // 添加你的程序的暂停逻辑 changes <- svc.Status{State: svc.Paused, Accepts: cmdsAccepted} case svc.Continue: // 添加你的程序的继续逻辑 changes <- svc.Status{State: svc.Running, Accepts: cmdsAccepted} default: log.Printf("unexpected control request #%d", c) } } changes <- svc.Status{State: svc.StopPending} return } func RunService() { err := svc.Run("tamper_proof_refactored", &myService{}) if err != nil { log.Fatal(err) } fmt.Println("服务已启动") }
启动后,直接终端查看下效果
写了一个服务状态判断,打印出的服务当前状态日志
写完后,就可以正常的服务启动和关闭了,下面是详细的服务启动/关闭指令
sc create MyService binPath= "C:\path\to\your\program\tamper_proof_XXXX.exe" //创建服务 net start MyService // 开启服务 net stop myserver // 暂停服务 sc delete myserver // 删除服务
参考文献: