目录

Gin 优雅关机

什么是优雅关机?

优雅关机就是服务端关机命令发出后不是立即关机,而是等待当前还在处理的请求全部处理完毕后再退出程序,是一种对客户端友好的关机方式。而执行 Ctrl+C 关闭服务端时,会强制结束进程导致正在访问的请求出现问题。

 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
  // +build go1.8

  package main

  import (
      "context"
      "log"
      "net/http"
      "os"
      "os/signal"
      "time"

      "github.com/gin-gonic/gin"
  )

  func main() {
      router := gin.Default()
      router.GET("/", func(c *gin.Context) {
          time.Sleep(5 * time.Second)
          c.String(http.StatusOK, "Welcome Gin Server")
      })

      srv := &http.Server{
          Addr:    ":8080",
          Handler: router,
      }

      go func() {
          // 服务连接
          if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
              log.Fatalf("listen: %s\n", err)
          }
      }()


      quit := make(chan os.Signal, 1) // 创建一个接收信号的通道
      // kill 默认会发送 syscall.SIGTERM 信号
      // kill -2 发送 syscall.SIGINT 信号,我们常用的Ctrl+C就是触发系统SIGINT信号
      // kill -9 发送 syscall.SIGKILL 信号,但是不能被捕获,所以不需要添加它
      // signal.Notify把收到的 syscall.SIGINT或syscall.SIGTERM 信号转发给quit
      signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)  // 此处不会阻塞
      <-quit  // 阻塞在此,当接收到上述两种信号时才会往下执行
      log.Println("Shutdown Server ...")
      // 创建一个5秒超时的context
      ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
      defer cancel()
      // 5秒内优雅关闭服务(将未处理完的请求处理完再关闭服务),超过5秒就超时退出
      if err := srv.Shutdown(ctx); err != nil {
        log.Fatal("Server Shutdown: ", err)
      }

      log.Println("Server exiting")
}