贝利信息

c# 在高并发服务中,如何做优雅停机(Graceful Shutdown)

日期:2026-01-18 00:00 / 作者:月夜之吻
GracefulShutdown 是指等待处理中请求完成、拒绝新请求、释放资源后再退出;ASP.NET Core 通过 IHostApplicationLifetime.ApplicationStopping 令牌协调,BackgroundService 需正确传递并响应 CancellationToken,清理操作必须异步且可控。

什么是 GracefulShutdown 在 C# 服务中的真实含义

它不是简单调用 Environment.Exit() 或杀进程,而是:等正在处理的请求完成、拒绝新请求、释放资源(如数据库连接、HttpClient 实例、后台任务)、再退出。在 ASP.NET Core 中,这由 IHostApplicationLifetimeIHostedService 协同控制;在裸 BackgroundService 或自托管场景中,需手动监听取消信号。

ASP.NET Core Web API 如何注册优雅停机逻辑

核心是利用 IHostApplicationLifetimeApplicationStopping 取消令牌,在其触发时做清理,同时确保中间件/控制器不接收新请求。关键点在于:HTTP 服务器(Kestrel)默认已支持 graceful shutdown,但你的业务逻辑必须响应取消信号。

var builder = WebApplication.CreateBuilder(args);
builder.Services.AddHostedService();

var app = builder.Build(); app.MapGet("/health", () => "ok");

// 注册停机钩子 app.Lifetime.ApplicationStopping.Register(() => { Console.WriteLine("Shutting down gracefully..."); // 这里可 await,但注意:ApplicationStopping.Token 已触发,不可再用于新操作 // 推荐把清理逻辑封装进 async 方法,并用 Task.Run + await 配合 Token _ = Task.Run(async () => { await MyCleanupAsync(app.Lifetime.ApplicationStopping.Token); }); });

await app.RunAsync();

BackgroundService 中如何正确响应停机信号

BackgroundService 本身已内置对 CancellationToken 的支持,但常见错误是忽略 StopAsync 的超时约束或未取消内部循环。

public class MyBackgroundWorker : BackgroundService
{
    private readonly ILogger _logger;
public MyBackgroundWorker(ILoggerzuojiankuohaophpcnMyBackgroundWorkeryoujiankuohaophpcn logger)
{
    _logger = logger;
}

protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
    while (!stoppingToken.IsCancellationRequested)
    {
        try
        {
            await DoWorkAsync(stoppingToken);
            await Task.Delay(5000, stoppingToken); // 注意:token 来自 ExecuteAsync 参数
        }
        catch (OperationCanceledException)
        {
            break;
        }
        catch (Exception ex)
        {
            _logger.LogError(ex, "Error in background worker");
        }
    }
}

private async Task DoWorkAsync(CancellationToken ct)
{
    // 所有 await 必须传入 ct
    await _httpCl

ient.GetAsync("https://api.example.com/health", ct); } public override async Task StopAsync(CancellationToken stoppingToken) { _logger.LogInformation("Stopping background worker..."); // 等待当前轮次完成,但不超过 stoppingToken 超时 await base.StopAsync(stoppingToken); }

}

容易被忽略的三个破坏点

哪怕代码写了 await ... CancellationToken,仍可能因以下原因导致“假优雅”:

  • HttpClient 实例未复用且未设置 Timeout:单次请求卡死会阻塞整个停机流程;建议统一用 IHttpClientFactory 并配置 DefaultRequestTimeout
  • 数据库连接池未释放:EF Core 的 SaveChangesAsync 若没传 token,可能永远挂住;DbContext 应注册为 Scoped,并在 StopAsync 后显式调用 DisposeAsync()
  • 第三方 SDK(如 Redis client、RabbitMQ consumer)未提供 cancellation 支持:必须查阅文档确认其 StopAsync 是否真正等待消息 ACK 完成,否则可能丢消息

最危险的是:你以为停机成功了,其实某个 Task.Run(() => { Thread.Sleep(30000); }) 还在跑 —— 它完全脱离取消体系,只能靠超时强杀。