Golang 平滑重启之优雅关机

前言

作为一个web服务,升级操作是一个必不可少的过程,但是在升级之前还有个操作,那就是:退出服务(关机),当然我们可以暴力的终止程序然后启动新服务,但是这是基于业务不敏感的情况下,正常我们需要让用户的一次请求完成之后才终止程序,幸运的是Go在1.8+上面增加shutdown方法可以很简单的实现此过程。


一、shutdown机制

先说明一下Shutdown工作的机制:当程序检测到中断信号时,我们调用http.server种的shutdown方法,该方法将阻止新的请求进来,同时保持当前的连接,知道当前连接完成则终止程序!

二、使用步骤

通过Gin提供的一个案例来解释说明

代码如下:

var i = 0
func main() {
	router := gin.Default()
	//创建两个接口,一个延迟9秒钟返回信息
	router.GET("/", func(c *gin.Context) {
		time.Sleep(9 * time.Second)
		i++
		c.JSON(http.StatusOK,gin.H{
			"num":i,
		})
	})
	//一个立刻返回信息
	router.GET("/a", func(c *gin.Context) {
		i++
		c.JSON(http.StatusOK,gin.H{
			"num":i,
		})
	})

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

	// Initializing the server in a goroutine so that
	// it won't block the graceful shutdown handling below
	go func() {
		if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
			log.Fatalf("listen: %s\n", err)
		}
	}()
	
	//创建一个信号监听通道
	quit := make(chan os.Signal, 1)
	//监听 syscall.SIGINT 跟 syscall.SIGTERM信号
	signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
	si :=<-quit
	log.Println("Shutting down server...",si)

	//shutdown方法需要传入一个上下文参数,这里就设计到两种用法
	//1.WithCancel带时间,表示接收到信号之后,过完该断时间不管当前请求是否完成,强制断开
	ctx, cancel := context.WithTimeout(context.Background(), 9*time.Second)
	//2.不带时间,表示等待当前请求全部完成再断开
	ctx, cancel := context.WithCancel(context.Background())
	defer cancel()
	if err := srv.Shutdown(ctx); err != nil {
		//当请求还在的时候强制断开了连接将产生错误,err不为空
		log.Fatal("Server forced to shutdown:", err)
	}
	log.Println("Server exiting")
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值