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

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

优雅停止

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

优雅启动

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

加载配置文件。

建立数据库连接池。

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

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

......

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

相关推荐
嘟嘟MD4 小时前
程序员副业 | 2025年9月复盘
后端·aigc
尘觉4 小时前
中秋节与 Spring Boot 的思考:一场开箱即用的团圆盛宴
java·spring boot·后端
间彧5 小时前
Seata分布式事务框架详解与项目实战
后端
zhuyasen5 小时前
单机已达上限?PerfTest 分布式压测登场,轻松模拟百万用户洪峰
后端·性能优化·测试
勇哥java实战分享5 小时前
sensitive-word:一个简单易用的敏感词过滤框架
后端
popoxf5 小时前
spring容器启动流程(反射视角)
java·后端·spring
Funcy6 小时前
XxlJob 源码08:任务执行流程(三)之执行器揭秘
后端
AAA修煤气灶刘哥6 小时前
监控摄像头?不,我们管这个叫优雅的埋点艺术!
java·后端·spring cloud
2301_772093566 小时前
tuchuang_后端_前端_注册登录
数据库·后端·网络协议·mysql·wireshark