Go HTTP 服务“优雅退出”有两个特点
- 程序中断退出时,需要给程序预留退出时间,等待处理中的请求处理完毕后再退出
- 程序中断退出时,客户端新发起的请求,会被服务器拦截,无法执行业务逻辑
实验前准备
- 编写一个接口服务。访问接口
/hello
,Sleep 10s,返回响应结果 - 将 server.ListenAndServe 放到一个 goroutine 协程中,如果返回错误不是
http.ErrServerClosed
(这里需要特别注意下),panic 退出 - 主线程监听系统中断指令,程序收到中断指令
syscall.SIGINT, syscall.SIGTERM
后,初始化一个 10s 超时取消的context.Context
- 将超时取消的
context.Context
作为参数传入server.Shutdown(ctx)
,程序等待 10s 关闭
package main
import (
"context"
"net/http"
"os"
"os/signal"
"syscall"
"time"
)
func main() {
server := &http.Server{
Addr: ":9034",
}
ctx := context.Background()
http.Handle("/hello", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
time.Sleep(time.Second * 10)
w.Write([]byte("hello\n"))
}))
go func() {
err := server.ListenAndServe()
if err != nil && err != http.ErrServerClosed {
panic(err)
}
}()
done := make(chan os.Signal, 1)
signal.Notify(done, syscall.SIGINT, syscall.SIGTERM)
<-done
ctx, cancel := context.WithTimeout(ctx, time.Second*10)
defer cancel()
if err := server.Shutdown(ctx); err != nil {
panic(err)
}
}
实验测试
- 终端 1 启动程序
go run main.go
- 终端 2 发起 CURL 请求
curl http://localhost:9034/hello
- 返回终端 1,Ctrl + C 结束程序
- 新启终端 3,发起 CURL 请求
curl http://localhost:9034/hello
实验结果
- 终端 1,10s 之后退出
- 终端 2 请求正常拿到返回结果
- 终端 3 请求失败
生产者、消费者的优雅退出
- 创建一个 channel,往 channel 中写入 1000 个数据
- 监听系统中断信号,收到中断信号后,关闭 channel,等待 channel 中的数据处理完毕后退出
package main
import (
"context"
"fmt"
"os"
"os/signal"
"syscall"
"time"
)
func main() {
queue := make(chan int64, 1000)
done := make(chan os.Signal, 1)
signal.Notify(done, syscall.SIGINT, syscall.SIGTERM)
end := 1000
for i := 0; i < end; i++ {
queue <- int64(i)
}
for {
select {
case i := <-queue:
fmt.Println(i)
if i == int64(end-1) {
return
}
time.Sleep(time.Microsecond * 100)
case <-done:
close(queue)
for i := range queue {
fmt.Println(i)
}
return
}
}
}
我第一次知道“优雅退出”这个名词来自 Golang 的相关知识,理解之后发现,其实这个名词并非 Golang 专属,任何程序都需要考虑给程序预留退出时间,以保证所有的逻辑处理完毕、数据完整等。
优雅退出的本质就是给进程留出清理的时间。程序退出时,在内存队列中可能有些数据还没有处理完,强制退出,会造成数据丢失的等问题;或者数据操作不被事务“包裹”,前面的数据处理逻辑已执行,流程中断,后面的部分数据未被处理,产生脏数据。
出现上面这些问题,在后期修复数据比较棘手,会出现不知道该修复哪些数据,或者无法修复的问题。可见“优雅退出”何其重要。