返回顶部
首页 > 资讯 > 后端开发 > GO >go如何优雅关闭Graceful Shutdown服务
  • 799
分享到

go如何优雅关闭Graceful Shutdown服务

go关闭服务Graceful Shutdowng 2023-05-20 08:05:20 799人浏览 八月长安
摘要

目录Shutdown方法ReGISterOnShutdown方法sync.WaitGroup处理退出函数Shutdown方法 Go1.8之后有了Shutdown方法,用来优雅的关闭(

Shutdown方法

Go1.8之后有了Shutdown方法,用来优雅的关闭(Graceful Shutdown)服务。

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

这个方法会在调用时进行下述操作:

1.关闭所有open listeners

2.关闭所有idle connections

3.无限期等待connections回归idle状态

4.之后关闭服务

注:3的无限期等待可以用context的超时来解决。

RegisterOnShutdown方法

Go1.9之后有了RegisterOnShutdown方法,用来在优雅的关闭服务的同时处理一些退出任务。

func (srv *Server) RegisterOnShutdown(f func())

关于RegisterOnShutdown如何生效,我们看代码。

type Server struct {
    ...
    onShutdown []func()
}

func (srv *Server) RegisterOnShutdown(f func()) {
    srv.mu.Lock()
    srv.onShutdown = append(srv.onShutdown, f)
    srv.mu.Unlock()
}

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

    srv.mu.Lock()
    ...
    for _, f := range srv.onShutdown {
        go f()
    }
    srv.mu.Unlock()

    ...
}

由代码可知,在通过RegisterOnShutdown进行注册之后,这些函数被放入一个函数切片中,并在Shutdown被调用的时候,为每个函数启动了一个goroutine进行处理。

但由于没有进行后续处理,于是这里有个小问题,Shutdown是不等待这些处理函数结束的,就可能比这些处理函数先完成并进而结束程序。

package main
import (
    "context"
    "log"
    "net/Http"
    "os"
    "os/signal"
    "time"
)
func main() {
    var srv http.Server
    srv.RegisterOnShutdown(func() {
        time.Sleep(3 * time.Second)
        log.Println("Exit func")
    })
    idleConnsClosed := make(chan struct{})
    go func() {
        sigint := make(chan os.Signal, 1)
        signal.Notify(sigint, os.Interrupt)
        <-sigint
        // We received an interrupt signal, shut down.
        if err := srv.Shutdown(context.Background()); err != nil {
            // Error from closing listeners, or context timeout:
            log.Printf("HTTP server Shutdown: %v", err)
        }
        close(idleConnsClosed)
    }()
    if err := srv.ListenAndServe(); err != http.ErrServerClosed {
        // Error starting or closing listener:
        log.Fatalf("HTTP server ListenAndServe: %v", err)
    }
    <-idleConnsClosed
}

运行时Ctrl+c之后,服务立刻结束了,并未等待log.Println("Exit func")的输出。

sync.WaitGroup处理退出函数

为了解决这个问题我们使用sync.WaitGroup来处理这些退出函数,于是我们有了下面的代码(解耦并增加了一些日志):

package main
import (
    "context"
    "log"
    "net/http"
    "os"
    "os/signal"
    "sync"
    "time"
)
func main() {
    var srv http.Server
    var wg sync.WaitGroup
    register := func(f func()) {
        wg.Add(1)
        log.Println("Exit func register")
        srv.RegisterOnShutdown(func() {
            defer wg.Done()
            f()
            log.Println("Exit func done")
        })
    }
    register(func() {
        time.Sleep(3 * time.Second)
        log.Println("Called on Shutdown")
    })
    idleConnsClosed := make(chan struct{})
    go func() {
        sigint := make(chan os.Signal, 1)
        signal.Notify(sigint, os.Interrupt)
        <-sigint
        // We received an interrupt signal, shut down.
        log.Println("Shutdown start")
        if err := srv.Shutdown(context.Background()); err != nil {
            // Error from closing listeners, or context timeout:
            log.Printf("HTTP server Shutdown: %v", err)
        }
        close(idleConnsClosed)
    }()
    if err := srv.ListenAndServe(); err != http.ErrServerClosed {
        // Error starting or closing listener:
        log.Fatalf("HTTP server ListenAndServe: %v", err)
    }
    <-idleConnsClosed
    log.Println("Shutdown finish")
    wg.Wait()
}

我们在RegisterOnShutdown时使用了time.Sleep(3 * time.Second)来模拟一个长时间的退出程序,并且在"Shutdown finish"之后进行等待,等待退出程序完成任务。

执行日志如下:

2023/05/12 16:48:53 Exit func register
^C2023/05/12 16:48:54 Shutdown start
2023/05/12 16:48:54 Shutdown finish
2023/05/12 16:48:57 Called on Shutdown
2023/05/12 16:48:57 Exit func done

可以看到这个服务本身很轻,Shutdown start之后立刻就finish并来到wg.Wait(),并等待register的函数完成之后再退出程序。

另外,如果在服务运行的过程中需要启动一些其他的任务,也可以使用

wg.Add(1)
go func() {
    defer wg.Done()
    ...
}

来保证任务在服务Shutdown之后可以顺利完成。

以上就是go如何优雅关闭Graceful Shutdown服务的详细内容,更多关于go关闭Graceful Shutdown的资料请关注编程网其它相关文章!

您可能感兴趣的文档:

--结束END--

本文标题: go如何优雅关闭Graceful Shutdown服务

本文链接: https://lsjlt.com/news/214215.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • go如何优雅关闭Graceful Shutdown服务
    目录Shutdown方法RegisterOnShutdown方法sync.WaitGroup处理退出函数Shutdown方法 Go1.8之后有了Shutdown方法,用来优雅的关闭(...
    99+
    2023-05-20
    go关闭服务 Graceful Shutdowng
  • go关闭Graceful Shutdown服务的几种方法
    本篇文章和大家了解一下go关闭Graceful Shutdown服务的几种方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。目录Shutdown方法RegisterOnShutdown方法sync.WaitGr...
    99+
    2023-07-06
  • Go如何优雅的关闭goroutine协程
    目录1.简介2.为什么需要关闭goroutine2.1 协程的生命周期2.2 协程的终止条件2.3 为什么需要主动关闭goroutine3.如何优雅得关闭goroutine3.1 传...
    99+
    2023-05-20
    Go关闭goroutine协程 Go关闭goroutine Go关闭协程
  • go实现服务优雅关闭的示例
    目录为什么需要优雅关闭监控服务退出信号拒绝新请求执行关闭之前的回调等待处理中的请求执行完毕实战支持强制退出为什么需要优雅关闭 什么叫优雅关闭?先说不优雅关闭,就是什么都不管,强制关闭...
    99+
    2023-02-09
    go 服务关闭
  • 如何在 Golang 中优雅地关闭 HTTP 服务器
    Golang (又称为 Go) 是一门相对比较年轻的编程语言,它被广泛应用于后端服务和 API 的开发中。而 HTTP 是一个常用的协议,很多 Golang 开发人员会选择使用 Golang 中自带的 HTTP 包来进行 HTTP 服务器的...
    99+
    2023-05-14
    Golang go语言 http
  • 解析golang中如何优雅地关闭http服务
    Golang是一种非常流行的编程语言,它具有高效的并发处理能力和优秀的性能表现。相信许多golang的开发人员都会遇到一个这样的问题,在golang中如何优雅地关闭http服务?首先,我们需要知道,创建一个http服务是比较容易的,只需要几...
    99+
    2023-05-14
  • 【Java】如何优雅的关闭线程池
    文章目录 背景一、线程中断 interrupt二、线程池的关闭 shutdown 方法2.1、第一步:advanceRunState(SHUTDOWN) 把线程池置为 SHUTDOWN2.2、...
    99+
    2023-09-21
    java 开发语言
  • go语言开发中如何优雅得关闭协程方法
    目录1.简介2.为什么需要关闭goroutine2.1 协程的生命周期2.2 协程的终止条件2.3 为什么需要主动关闭goroutine3.如何优雅得关闭goroutine3.1 传...
    99+
    2023-05-20
    go得到关闭协程 go关闭协程
  • golang中怎么优雅地关闭http服务
    这篇文章主要介绍“golang中怎么优雅地关闭http服务”,在日常操作中,相信很多人在golang中怎么优雅地关闭http服务问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”golang中怎么优雅地关闭htt...
    99+
    2023-07-06
  • 如何优雅、安全的关闭MySQL进程
    前言 本文分析了 mysqld 进程关闭的过程,以及如何安全、缓和地关闭 MySQL 实例,对这个过程不甚清楚的同学可以参考下。 关闭过程 发起 shutdown,发出 SIGTERM 信号 有必要的话,新建一个关闭...
    99+
    2022-05-12
    MySQL 进程 MySQL 关闭进程
  • 怎么在Golang中优雅地关闭HTTP服务器
    这篇“怎么在Golang中优雅地关闭HTTP服务器”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“怎么在Golang中优雅地关...
    99+
    2023-07-05
  • go语言开发优雅得关闭协程的方法
    本篇文章和大家了解一下go语言开发优雅得关闭协程的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。1.简介本文将介绍首先为什么需要主动关闭goroutine,并介绍如何在Go语言中关闭goroutine的常见套路,包括...
    99+
    2023-07-06
  • python多线程编程:如何优雅地关闭线程
    在并发编程中,我们可能会创建新线程,并在其中运行任务,可能由于一些原因,决定停止该线程。例如: 不再需要线程任务的结果了。应用程序正在关闭。线程执行可能已经出现了异常 关于python多线程编程知...
    99+
    2023-09-04
    python 开发语言
  • MySQL shutdown unexpectedly - 如何解决MySQL报错:MySQL意外关闭
    当MySQL意外关闭时,我们首先应该查看MySQL的错误日志,以了解关闭的原因。通常,MySQL的错误日志位于MySQL安装目录的data文件夹下,默认为mysql-error.log。我们可以使用文本编辑器打开日志文件,查找最近的错误信息...
    99+
    2023-10-21
    MySQL shutdown 解决方法
  • 如何关闭Sendmail服务
    这篇文章给大家分享的是有关如何关闭Sendmail服务的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。关闭sendmail服务(1)[root@sample ~]# /etc/rc.d/init.d/sendmai...
    99+
    2023-06-16
  • go中如何关闭http端口?
    在Go语言中,关闭HTTP端口可以通过使用`http.Server`的`Close()`方法来实现。首先,我们需要创建一个`http.Server`实例并监听指定的端口。一旦服务器开始...
    99+
    2024-02-09
    go语言
  • centos7如何关闭mongodb服务
    centos7中关闭mongodb服务的方法:1、打开centos7终端;2、输入“systemctl start mongod.service”命令关闭mongodb服务即可。具体操作步骤:在centos7系统桌面中使用快捷键【Ctrl+...
    99+
    2024-04-02
  • ubuntu如何关闭mongodb服务
    ubuntu中关闭mongodb服务的方法:1、打开ubuntu终端;2、输入“sudo service mongod stop”命令停止mongodb服务即可。具体操作步骤:在ubuntu系统桌面中使用快捷键【Ctrl+Alt+T】打开u...
    99+
    2024-04-02
  • linux如何关闭mongodb服务
    linux中关闭mongodb服务的方法:1、打开linux终端;2、在终端命令行模式下输入“cd /usr/local/src/mongodb/bin”命令切换至本地mongodb目录下的bin文件夹;3、最后可输入“./mongod -...
    99+
    2024-04-02
  • centos7如何关闭redis服务
    centos7中关闭redis服务的方法:1、打开centos7终端;2、输入“./redis-cli -p 6379 shutdown”命令关闭redis服务即可。具体操作步骤:在centos7系统桌面中使用快捷键【Ctrl+Alt+T】...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作