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

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

优雅停止

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),它会立即关闭它们

优雅启动

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

加载配置文件。

建立数据库连接池。

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

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

......

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

相关推荐
小村儿1 小时前
连载04-最重要的Skill---一起吃透 Claude Code,告别 AI coding 迷茫
前端·后端·ai编程
IT_陈寒2 小时前
Vite的alias配置把我整不会了,原来是这个坑
前端·人工智能·后端
gelald2 小时前
Spring Boot - 自动配置原理
java·spring boot·后端
希望永不加班2 小时前
SpringBoot 集成测试:@SpringBootTest 与 MockMvc
java·spring boot·后端·log4j·集成测试
uzong2 小时前
软件人员可以关注的 Skill,亲测确实不错,值得试一下
人工智能·后端
掘金虾2 小时前
Hono 框架入门到实战:用 Node.js 写一个支持工具调用的流式对话 Agent
后端
用户8356290780512 小时前
Python 自动拆分 Word 文档教程:按分节符与分页符处理
后端·python
树獭叔叔3 小时前
Claude Code 工具系统深度剖析:从静态注册到动态发现
后端·aigc·openai
树獭叔叔3 小时前
Claude Code 的上下文管理:多层渐进式压缩架构深度解析
后端·aigc·openai
计算机学姐3 小时前
基于SpringBoot的高校竞赛管理系统
java·spring boot·后端·spring·信息可视化·tomcat·mybatis