优雅的关闭golang server
每当服务由于各种原因需要关闭时,常见的是操作系统中断,我们希望我们的服务能够优雅地关闭。
我们希望我们的golang服务停止接收新的请求,同时完成正在进行的请求并返回其响应,然后最终关闭。
我们可以通过创建一个带有cancel()回调函数的context对象和http.Server接口提供的关闭方法来实现这一任务。 通过一个channel持续监听操作系统的中断,然后调用context的cancel()函数。
我们使用该context创建一个服务器实例,然后在context完成后关闭服务。
package main
import (
"context"
"fmt"
"log"
"net/http"
"os"
"os/signal"
"time"
)
func serve(ctx context.Context) (err error) {
mux := http.NewServeMux()
mux.Handle("/", http.HandlerFunc(
func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, "okay")
},
))
srv := &http.Server{
Addr: ":6969",
Handler: mux,
}
go func() {
if err = srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
log.Fatalf("listen:%+s\n", err)
}
}()
log.Printf("server started")
<-ctx.Done()
log.Printf("server stopped")
ctxShutDown, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer func() {
cancel()
}()
if err = srv.Shutdown(ctxShutDown); err != nil {
log.Fatalf("server Shutdown Failed:%+s", err)
}
log.Printf("server exited properly")
if err == http.ErrServerClosed {
err = nil
}
return
}
func main() {
c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt)
ctx, cancel := context.WithCancel(context.Background())
go func() {
oscall := <-c
log.Printf("system call:%+v", oscall)
cancel()
}()
if err := serve(ctx); err != nil {
log.Printf("failed to serve:+%v\n", err)
}
}
本文由 络壳 原创或整理,转载请注明出处