c# 如何提高Web API的并发能力 c# API性能调优

来源:这里教程网 时间:2026-02-21 17:36:01 作者:

async
/
await
替换同步 I/O 操作

Web API 的并发瓶颈常来自阻塞式 I/O,比如

HttpClient.GetStringAsync
未被正确 await、或直接调用
File.ReadAllText
这类同步文件读取。这些操作会占用线程池线程,导致高并发下线程耗尽、请求排队。

实操建议:

所有外部调用(HTTP、数据库、文件、缓存)必须使用异步版本,并完整 await,避免
.Result
.Wait()
ASP.NET Core 默认配置下,
ThreadPool.SetMinThreads(100, 100)
这类手动调优无效且危险——.NET 6+ 已自动优化线程池增长策略,强行修改反而干扰调度
检查第三方 SDK 是否提供异步 API;若只有同步接口(如某些旧版 Redis 客户端),考虑升级或封装为
Task.Run
(仅限 CPU-bound 场景,I/O-bound 不适用)
public async Task<ActionResult<User>> Get(int id)
{
    // ✅ 正确:全程 async/await
    var user = await _userRepository.GetByIdAsync(id);
    if (user == null) return NotFound();
<pre class='brush:php;toolbar:false;'>var profile = await _httpService.GetProfileAsync(user.ProfileUrl);
return Ok(new { user, profile });

}

减少中间件和过滤器中的同步阻塞逻辑

自定义

ActionFilter
AuthorizationHandler
或中间件里执行
HttpContext.Request.Body.Read
JsonConvert.DeserializeObject
等操作,极易成为并发短板。尤其当请求体较大时,同步反序列化会锁住线程。

实操建议:

避免在
OnActionExecuting
中同步读取
Request.Body
;改用
HttpContext.Request.ReadFormAsync()
或模型绑定(Model Binding)自动处理
日志记录不要在过滤器中做耗时格式化(如拼接大量字符串或调用外部服务),改用结构化日志(Serilog +
LogContext
)并异步写入
身份验证尽量复用
Bearer
+ JWT 验证,避免每次请求都查数据库;必要时加内存缓存(
IDistributedCache
)存 token 声明

合理配置 Kestrel 和连接管理

Kestrel 是 ASP.NET Core 默认服务器,其默认连接限制和缓冲区设置在高并发场景下可能成为隐性瓶颈。例如未调整

MaxConcurrentConnections
MaxRequestBodySize
,会导致连接被静默拒绝或内存暴涨。

实操建议:

Program.cs
中显式配置 Kestrel:
options.Limits.MaxConcurrentConnections = 1000;
(根据服务器核数和内存调整,非越大越好)
禁用不必要的 HTTP/2 特性(如服务器推送)——除非明确需要,否则它会增加连接状态开销 对上传接口单独设限:
options.Limits.MaxRequestBodySize = 10 * 1024 * 1024;
,避免大文件请求长期占用连接
确认反向代理(Nginx / Azure Front Door)未设置过低的
keepalive_timeout
或连接数限制,否则 Kestrel 调优无效
var builder = WebApplication.CreateBuilder(args);
builder.WebHost.ConfigureKestrel(serverOptions =>
{
    serverOptions.Limits.MaxConcurrentConnections = 500;
    serverOptions.Limits.MaxRequestBodySize = 5 * 1024 * 1024;
});

慎用
Task.Run
和同步上下文切换

看到“CPU 密集型任务”就套

Task.Run
是常见误区。它只是把工作扔给线程池,但若大量使用,反而加剧线程竞争,且无法解决 I/O 瓶颈。

实操建议:

仅在真正 CPU-bound 场景下用
Task.Run
(如图像缩放、加密解密、复杂计算),并确保有明确超时和取消支持
避免在 Controller 中写
Task.Run(() => { /* 同步 DB 查询 */ }).Result
——这等于用线程池线程干了本该由异步驱动的事,还丢了上下文
不要在
async void
方法中处理请求逻辑(如事件处理器),会导致异常无法被捕获,影响整个请求生命周期

实际压测时,

HttpClient
复用、数据库连接池大小、JSON 序列化器配置(如禁用
ReferenceHandler
循环引用检测)往往比代码结构更影响吞吐量。这些细节不显眼,但一并发就暴露。

相关推荐