7

很高兴Go 1.8发布了,这是个值得庆祝的日子。

如何优雅的关闭http服务在Go Web开发中一直被提及和讨论的话题,今天Go 1.8的发布终于为我们带来了这个特性。

文档中是这样介绍的:

func (srv *Server) Shutdown(ctx context.Context) error

Shutdown 将无中断的关闭正在活跃的连接,然后平滑的停止服务。处理流程如下:

  • 首先关闭所有的监听

  • 然后关闭所有的空闲连接

  • 然后无限期等待连接处理完毕转为空闲,并关闭

  • 如果提供了 带有超时的Context,将在服务关闭前返回 Context的超时错误

需要注意的是,Shutdown 并不尝试关闭或者等待 hijacked连接,如 WebSockets。如果需要的话调用者需要分别处理诸如长连接类型的等待和关闭。

其实,你只要调用 Shutdown 方法就好了。

简单示例:

// main.go

package main

import (
    "fmt"
    "log"
    "net/http"
    "os"
    "os/signal"
    "syscall"
    "time"
)

func main() {
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        fmt.Fprintf(w, "Hello World, %v\n", time.Now())
    })

    s := &http.Server{
        Addr:           ":8080",
        Handler:        http.DefaultServeMux,
        ReadTimeout:    10 * time.Second,
        WriteTimeout:   10 * time.Second,
        MaxHeaderBytes: 1 << 20,
    }

    go func() {
        log.Println(s.ListenAndServe())
        log.Println("server shutdown")
    }()

    // Handle SIGINT and SIGTERM.
    ch := make(chan os.Signal)
    signal.Notify(ch, syscall.SIGINT, syscall.SIGTERM)
    log.Println(<-ch)

    // Stop the service gracefully.
    log.Println(s.Shutdown(nil))

    // Wait gorotine print shutdown message
    time.Sleep(time.Second * 5)
    log.Println("done.")
}

运行程序:

go run main.go

然后 ctrl + c 终止该程序,会打印出如下信息:

2017/02/17 11:36:28 interrupt
2017/02/17 11:36:28 <nil>
2017/02/17 11:36:28 http: Server closed
2017/02/17 11:36:28 server shutdown

可以看到,服务被正确关闭了。

在没有 Shutdown方法之前,ctrl + c就硬生生的终止了,然后就没有然后了。

更多内容见官方文档:

https://golang.org/pkg/net/ht...


hengfeiyang
1.7k 声望290 粉丝