服务优雅停止和服务优雅启动

问题请在评论区中一起交流,一起进步

优雅停止

1.什么时候会触发优雅停止?

像突然停电,硬件物理损坏,进程收到sigkill信号(执行kill -9 <PID>)时,服务会被立即终止,不会触发优雅停止

当服务在滚动更新版本(master从1.2.1升级到1.2.2),服务缩容(根据负载情况减少实例数量),服务器关机或者重启时,会触发优雅停止,即执行一系列善后工作:

1.停止接收新的请求。

2.等待当前正在处理的请求全部完成。

3.释放占用的资源,如数据库连接、文件句柄等。

4.执行清理工作,如将内存中的数据刷到磁盘。

5.所有工作完成后,再安全退出。

2.如果不进行优雅停止,会发生什么?比如:

一个请求刚处理到一半(比如,数据库事务只提交了一部分),服务突然被终止 这会导致:

1.数据不一致:用户的订单可能只创建了一半。

2.客户端错误:用户会看到一个突然断开的连接错误。

3.资源泄露:数据库连接可能没有被正常关闭。

3.在go中如何实现:

Go 的 net/http 包从 1.8 版本开始,为 http.Server 提供了原生的 Shutdown() 方法,让实现优雅停机变得非常简单

Go 复制代码
func main() {
    // 创建一个channel用于接收系统信号
    quitChannel := make(chan os.Signal, 1)

    //将操作系统发来的 os.Interrupt 信号(通常是用户按下了 Ctrl+C)转发到 quitChannel 这个通道里。
    signal.Notify(quitChannel, os.Interrupt)

    //创建一个http请求路由器
    serverMux := http.NewServeMux()
    // 注册路由
    serverMux.HandleFunc("/", func(writer http.ResponseWriter, request *http.Request) {
       log.Println("Received a new request...")
       writer.Write([]byte("Hello, this is a graceful shutdown example."))
       log.Println("Finished processing request.")
    })

    server := &http.Server{
       Addr:    ":8080",
       Handler: serverMux,
    }

    // 起个goroutine来启动http服务
    go func() {
       log.Println("Server is starting on port 8080...")
       if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
          log.Fatalf("Could not listen on %s: %v\n", server.Addr, err)
       }
    }()

    // 阻塞主 goroutine,直到接收到关闭信号
    <-quitChannel
    log.Println("Received shutdown signal. Server is shutting down...")
    
    // 收到关闭信号后,开始优雅关机
    
    // 设置超时时间,超出这个时间,即使请求还是执行,也要关闭这个连接了
    ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
    defer cancel()

    // 调用 Shutdown(),开始优雅关闭
    // Shutdown 会阻塞,直到所有连接都处理完毕或 context 超时
    if err := server.Shutdown(ctx); err != nil {
       log.Fatalf("Server shutdown failed: %v", err)
    }

    log.Println("Server has been shut down gracefully.")
}

Shutdown() 方法原理就是:

1.关闭http服务器的监听端口(任何新的客户端请求都无法建立连接,它们会立刻被拒绝)

2.会遍历当前服务器持有的所有连接,

【1】对于正在处理请求的连接 (Active Connections),它会耐心地等待它们完成当前的请求-响应周期,注意此时会进行超时强行关停连接,即请求执行时间太长就会强行关闭连接,比如设置10s后就要强行终止连接了

【2】对于已经处理完请求但保持着 keep-alive 的空闲连接 (Idle Connections),它会立即关闭它们

优雅启动

服务在宣布"我已经准备好对外提供服务"之前,必须完成所有必要的初始化工作。这包括:

加载配置文件。

建立数据库连接池。

初始化缓存(缓存预热)。

连接到其他依赖的微服务。

......

只有当所有依赖都准备就绪后,服务才开始监听端口,接收外部流量。

相关推荐
GetcharZp3 小时前
玩转 Linux 机器视觉:手把手带你搞定 Ubuntu 下海康工业相机 C++ SDK
后端
星星在线6 小时前
MusicFree:一个「All in One」的个人音乐服务器,让听歌回归简单
前端·后端
IT_陈寒7 小时前
Redis的SETNX并发问题让我加了三天班
前端·人工智能·后端
demo007x7 小时前
Docling 文档转换以及技术架构分析
前端·后端·程序员
袋鱼不重9 小时前
我的神奇同事,AI 用多了居然写了个 Open In Codex
前端·后端·ai编程
用户8356290780519 小时前
使用 Python 操作 Word 内容控件
后端·python
像我这样帅的人丶你还9 小时前
啥? 前端也要会干Java?🛵🛵🛵
后端
Hommy889 小时前
【剪映小助手】添加贴纸接口(Add Sticker)
后端·github·剪映小助手·视频剪辑自动化·剪映api
CaffeinePro9 小时前
FastAPI响应处理:返回值、状态码、响应头与异常标准化与案例解析
后端
HuanYu10 小时前
PageHelper分页的原理
后端