每当服务由于各种原因需要关闭时,常见的是操作系统中断,我们希望我们的服务能够优雅地关闭。
我们希望我们的golang服务停止接收新的请求,同时完成正在进行的请求并返回其响应,然后最终关闭。
我们可以通过创建一个带有cancel()回调函数的context对象和http.Server接口提供的关闭方法来实现这一任务。
通过一个channel持续监听操作系统的中断,然后调用context的cancel()函数。
我们使用该context创建一个服务器实例,然后在context完成后关闭服务。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
| 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) } }
|