当前位置: 首页 > news >正文

12 | 给应用添加优雅关停功能

提示:

  • 所有体系课见专栏:Go 项目开发极速入门实战课;
  • 本节课最终源码位于 fastgo 项目的 feature/s09 分支。

在成功实现一个简单的 Web 服务器之后,接下来需要进一步完善其核心功能,使其更贴合实际需求并满足企业级应用场景的要求。例如,需要实现功能丰富的中间件、处理跨域访问、以及支持优雅关停等功能。本节课将深入介绍这些关键功能的实现,帮助我们打造一个更加健壮、高效、且易于维护的 Web 服务器。

添加优雅关停功能

Web 服务器通常都需要实现优雅关停功能。优雅关停服务器可以带来很多好处,例如提高 API 接口的成功率,减少系统脏数据的出现概率等。本节会详细介绍如何实现优雅关停功能。

优雅关停的必要性

在应用程序的生命周期中,新功能发布、缺陷修复、配置变更等操作都需要重启服务。在服务进程停止时,可能需要执行一些必要的处理工作,例如:

  • 正在执行的 HTTP 请求需要等待其完成并返回结果,否则可能会导致请求报错或产生脏数据;
  • 异步处理任务需要将缓存中的数据处理完成,否则可能会导致数据丢失或不一致;
  • 关闭数据库连接,否则数据库连接池可能会保留无效连接,浪费宝贵的连接资源。

为了解决上述问题,建议的做法是给应用添加优雅关停功能,以提高系统的健壮性。

优雅关停的实现思路

实现优雅关停的最佳实践是在服务进程停止前,等待所有任务处理完成后再退出进程。在 Go 应用开发中,可以通过向应用程序发送标准的系统信号来终止服务进程。以下是最常见的三种终止方式:

  • CTRL+C:发送 SIGINT 信号;
  • kill <pid>:向指定进程发送 SIGTERM 信号;
  • kill -9 <pid>:向指定进程发送 SIGKILL 信号,强制终止信号。需要注意的是,SIGKILL 信号既不能被应用程序捕获,也不能被阻塞或忽略。

提示:在日常关闭服务时,应尽量避免使用 kill -9 命令,因为此命令会导致应用进程无法执行优雅关停逻辑。

fastgo 实现优雅关停功能

如果能够捕获 SIGINTSIGTERM 信号,并在捕获后执行一些关停逻辑,就可以实现优雅关停功能。实际上,当前 Go 应用的优雅关停功能大多基于这一思路实现。

Go 语言提供了 os/signal 包,用于监听并处理接收到的信号。基于上述思路,我们可以通过 os/signal 实现优雅关停功能,代码如下:

// 启动一些非阻塞任务,例如:HTTP / gRPC 服务,异步任务处理等逻辑// 创建一个 os.Signal 类型的 channel,用于接收系统信号
quit := make(chan os.Signal, 1)
// 当执行 kill 命令时(不带参数),默认会发送 syscall.SIGTERM 信号
// 使用 kill -2 命令会发送 syscall.SIGINT 信号(例如按 CTRL+C 触发)
// 使用 kill -9 命令会发送 syscall.SIGKILL 信号,但 SIGKILL 信号无法被捕获,因此无需监听和处理
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
// 阻塞程序,等待从 quit channel 中接收到信号
<-quit// 执行一些清理工作// 程序自然退出

上述代码执行优雅关停的流程如下:

  1. 启动 HTTP/gRPC 服务或其他异步任务,以非阻塞方式运行。如果服务本身是阻塞方式,可以通过 Go 协程启动;
  2. 创建一个 os.Signal 类型的 channel,用于捕获应用程序的关停信号;
  3. 调用 signal.Notify 函数,设置需要捕获的信号类型,例如 syscall.SIGINTsyscall.SIGTERM
  4. 使用 <-quit 阻塞主程序,等待信号到来;
  5. 当系统接收到 SIGINTSIGTERM 信号时,会向 quit 通道写入一条 os.Signal 类型的数据;
  6. quit 读取到信号后,解除阻塞状态,执行后续清理工作。清理完成后,进程正常退出。清理工作可根据业务逻辑执行不同的操作,例如通过 net/http 包的 Shutdown 方法优雅关闭 HTTP 服务。

在 Go 项目开发中,还可以通过一些 Go 包,例如:fvbock/endless 来实现优雅关停,但更推荐上面的方式,简单,并且不需要引入新包。很多优秀的开源项目,例如 Kubernetes 优雅关停实现思路跟上述思路保持一致。

fastgo 根据以上优雅关停功能实现思路,实现了优雅关停逻辑,代码如下(位于 internal/apiserver/server.go 文件中):

// Run 运行应用.
func (s *Server) Run() error {// 运行 HTTP 服务器// 打印一条日志,用来提示 HTTP 服务已经起来,方便排障slog.Info("Start to listening the incoming requests on http address", "addr", s.cfg.Addr)go func() {if err := s.srv.ListenAndServe(); err != nil && !errors.Is(err, http.ErrServerClosed) {slog.Error(err.Error())os.Exit(1)}}()// 创建一个 os.Signal 类型的 channel,用于接收系统信号quit := make(chan os.Signal, 1)// 当执行 kill 命令时(不带参数),默认会发送 syscall.SIGTERM 信号// 使用 kill -2 命令会发送 syscall.SIGINT 信号(例如按 CTRL+C 触发)// 使用 kill -9 命令会发送 syscall.SIGKILL 信号,但 SIGKILL 信号无法被捕获,因此无需监听和处理signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)// 阻塞程序,等待从 quit channel 中接收到信号<-quitslog.Info("Shutting down server ...")// 优雅关闭服务ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)defer cancel()// 先关闭依赖的服务,再关闭被依赖的服务// 10 秒内优雅关闭服务(将未处理完的请求处理完再关闭服务),超过 10 秒就超时退出if err := s.srv.Shutdown(ctx); err != nil {slog.Error("Insecure Server forced to shutdown", "err", err)return err}slog.Info("Server exited")return nil
}

在上述代码中,quit 收到 SIGINTSIGTERM 信号后,程序会解除阻塞状态,并调用 *http.Server类型实例的 Shutdown 方法优雅关停服务器。

通过 context.WithTimeout 创建上下文对象 ctx,其主要作用是为优雅关闭服务提供超时控制,确保服务在一定时间内完成清理工作。如果超过指定时间(这里是 10 秒),服务将被强制终止。ctx 被传递给 s.srv.Shutdown(ctx) 方法,用于通知服务相关的协程或其他子任务,当前服务正在关闭,并提供一个超时时间。服务中的任务可以通过监听 ctx.Done() 来检测是否需要终止,从而及时结束任务,避免资源泄漏。

*http.Server 类型的 Shutdown 方法的工作流程如下:首先关闭所有已开启的监听器,然后关闭所有空闲连接,最后等待所有活跃连接进入空闲状态后终止服务。如果传入的 ctx 在服务完成终止之前超时,则 Shutdown 方法会返回与 context 相关的错误。否则会返回由关闭服务监听器引发的其他错误。

Shutdown 方法被调用时,ServeListenAndServe 以及 ListenAndServeTLS 方法会立即返回 ErrServerClosed 错误。ErrServerClosed 错误被视为服务关闭时的正常行为。因此,如果 ListenAndServe 返回该错误,程序不会打印错误信息。

编译并测试优雅关停功能

执行以下命令,启动 fg-apiserver 服务:

$ ./build.sh
$ _output/fg-apiserver -c configs/fg-apiserver.yaml
time=2025-03-08T10:06:37.866+08:00 level=INFO msg="Start to listening the incoming requests on http address" addr=0.0.0.0:6666
^Ctime=2025-03-08T10:06:38.805+08:00 level=INFO msg="Shutting down server ..."
time=2025-03-08T10:06:38.806+08:00 level=INFO msg="Server exited"

在启动服务后,键入 CTRL+C,可以看到服务优雅退出日志。


欢迎加入「云原生AI 实战营」星球,12+ 高质量体系课、20+ 高质量实战项目助你在 AI 时代建立技术竞争力:
Alt文本


http://www.mrgr.cn/news/94159.html

相关文章:

  • 完整项目案例:基于Django的毕业设计选题管理系统(包含源码结构、核心代码及设计文档框架)
  • 【Linux】在VMWare中安装Ubuntu操作系统(2025最新_Ubuntu 24.04.2)#VMware安装Ubuntu实战分享#
  • Node-RED基础1
  • 【Godot4.3】RenderingServer总结
  • Quantum Computing:量子计算如何改变世界
  • 深入探索 Java Stream
  • 使用 OptiSLang 和 MotorCAD 构建一个强大的电机优化元模型
  • 感觉自己邮电部诗人
  • 【医院成本核算专题】8.大数据与医院成本核算的关联点:开启医疗成本管理新时代
  • 使用Dockerfile构建一个Docker镜像
  • 20 | 如何添加单元测试用例
  • 计算机:基于深度学习的Web应用安全漏洞检测与扫描
  • 工作记录 2017-01-06
  • 重要!!! 改进 梯度方差(Fisher 信息近似) 指数移动平均
  • 记录一下返修
  • 【操作系统】Linux基本命令2
  • JAVA SE 4.Java各版本特性
  • 【MapSet】哈希表
  • vue3,Element Plus中el-select默认显示0
  • 前端高阶面试题·每日一题