在 Go 1.16 的更新中,signal
包减少了一个函数 NotifyContext,
这让咱们优雅的重启服务(Graceful Restart)能够写的更加优雅。
一个服务想要优雅的重启次要蕴含两个方面:
- 退出的旧服务须要
Graceful Shutdown
,不强制杀过程,不透露系统资源。 - 在一个集群内轮流重启服务实例,保障服务不中断。
第二个问题跟部署形式相干,改天专门写一篇探讨,明天咱们次要谈怎么样优雅的退出。
首先在代码里,用了内部资源,肯定要应用 defer
去调用 Close()
办法敞开。
而后咱们就要拦挡零碎的中断信号,保障程序收到中断信号之后,被动有序退出,这样所有的 defer 才会被执行。
在以前,大略是这么写:
func everLoop(ctx context.Context) {
LOOP:
for {
select {case <-ctx.Done():
// 收到信号退出有限循环
break LOOP
default:
// 用一个 sleep 模仿业务逻辑
time.Sleep(time.Second * 10)
}
}
}
func main() {
// 建设一个能够手动勾销的 Context
ctx, cancel := context.WithCancel(context.Background())
// 监控零碎信号,这里只监控了 SIGINT(Ctrl+c),SIGTERM
// 在 systemd 和 docker 中,都是先发 SIGTERM,过一段时间没退出再发 SIGKILL
// 所以这里没捕捉 SIGKILL
sig := make(chan os.Signal, 1)
signal.Notify(sig, syscall.SIGINT, syscall.SIGTERM)
go func() {
<-sig
cancel()}()
// 开始有限循环,收到信号就会退出
everLoop(ctx)
fmt.Println("graceful shuwdown")
}
当初有了新的函数,这一段变得更简略了:
func main() {
// 监控零碎信号和创立 Context 当初一步搞定
ctx, stop := signal.NotifyContext(context.Background(), syscall.SIGINT, syscall.SIGTERM)
// 在收到信号的时候,会主动触发 ctx 的 Done,这个 stop 是不再捕捉注册的信号的意思,算是一种开释资源。defer stop()
// 开始有限循环,收到信号就会退出
everLoop(ctx)
fmt.Println("graceful shuwdown")
}
感激 Golang
,当年用别的语言须要写一大堆代码的性能,当初几行就能够轻松实现了。
让它成为你服务程序的标配吧。
最初,我是写最新的独立我的项目 LetServerRun 的时候,发现这种最新的写法的。
LetServerRun 能够让你把微信公众号当作随身的 Terminal 管制你的服务端。
在它的 Agent 的 main 函数中就有上述用法的示例,
欢送参考。
附上 LetServerRun 的服务号二维码,感兴趣的同学能够关注一下: