0

0

c# Flurl.Http 和 HttpClient 在并发请求中的易用性对比

月夜之吻

月夜之吻

发布时间:2026-01-19 12:20:02

|

703人浏览过

|

来源于php中文网

原创

Flurl.Http 默认不共享连接池,高并发易耗尽端口;应复用 FlurlClient 或全局配置 DefaultHttpClientFactory,并避免 new HttpClient(),推荐使用 IHttpClientFactory 管理生命周期。

c# flurl.http 和 httpclient 在并发请求中的易用性对比

Flurl.Http 的 GetAsync 默认不共享连接池,高并发下容易耗尽端口

Flurl.Http 默认为每个 FlurlClient 实例创建独立的 HttpClient,而它内部又默认启用 HttpMessageHandler 的新实例(即非复用)。这意味着每发起一个请求,都可能新建一个底层 TCP 连接,尤其在短生命周期、高频调用场景(如循环发 100 个 GetAsync)中,极易触发 SocketException: An operation on a socket could not be performed because the system lacked sufficient buffer space or because a queue was full

解决方法是显式复用同一个 FlurlClient 实例,并配置其 WithSettings 启用连接池:

var client = new FlurlClient()
    .WithSettings(s => {
        s.HttpClientFactory = new DefaultHttpClientFactory();
        // 确保复用底层 HttpClient 实例
    });

await "https://api.example.com/data".WithClient(client).GetAsync();

更稳妥的做法是全局注册单例 FlurlClient,避免每次构造:

  • 在启动时注册:FlurlHttp.Configure(c => c.HttpClientFactory = new DefaultHttpClientFactory());
  • 或直接复用静态 FlurlClient.Default(注意线程安全,它本身是线程安全的)

HttpClient 必须手动管理生命周期,直接 new HttpClient() 是反模式

很多人写 new HttpClient() 发请求,看似简单,实则埋雷:DNS 变更不生效、TIME_WAIT 连接堆积、SOCKET 耗尽。.NET 官方明确要求 HttpClient 应长期复用(如作为单例或注入 IHttpClientFactory)。

正确姿势只有两种:

  • 使用 IHttpClientFactory(推荐,尤其在 ASP.NET Core 中):
    services.AddHttpClient("myapi", c => {
        c.BaseAddress = new Uri("https://api.example.com/");
    });
    然后通过 IHttpClientFactory.CreateClient("myapi") 获取——它自动处理连接池、DNS 刷新、超时隔离
  • 手动单例复用(仅限极简控制台程序):
    private static readonly HttpClient _sharedClient = new HttpClient();
    但需自行配置 TimeoutDefaultRequestHeaders 等,且无法按命名区分策略

Flurl.Http 的链式语法在并发组合请求时更直观,但掩盖了错误传播细节

比如并发拉取多个用户信息:

北极象沉浸式AI翻译
北极象沉浸式AI翻译

免费的北极象沉浸式AI翻译 - 带您走进沉浸式AI的双语对照体验

下载
var tasks = userIds.Select(id => 
    $"https://api.example.com/users/{id}".GetJsonAsync());
var users = await Task.WhenAll(tasks);

这段代码比等价的 HttpClient 版本少写 5–6 行,可读性高。但它默认把所有异常包装成 FlurlHttpException,且 Task.WhenAll 遇到任一失败就整体抛出 AggregateException,原始 HTTP 状态码(如 429、503)需要从 ex.Call.Response.StatusCode 里挖。

而原生 HttpClient 更“透明”:

  • 你可以直接检查 HttpResponseMessage.IsSuccessStatusCode
  • EnsureSuccessStatusCode() 或手动 switch (res.StatusCode) 分流处理
  • 对 429 做重试、对 5xx 做降级,逻辑更可控

Flurl.Http 的默认超时是 100 秒,HttpClient 是无穷(TimeSpan.MaxValue)

这个差异在长轮询或大文件下载场景中会暴露问题:Flurl.Http 的 GetAsync 若卡住 100 秒自动中断并抛 TaskCanceledException;而裸 HttpClient 可能无限挂起(除非你显式设 Timeout 或用 CancellationToken)。

调整方式:

  • Flurl.Http 全局改:FlurlHttp.Configure(c => c.Timeout = TimeSpan.FromSeconds(30));
  • Flurl.Http 单次改:url.WithTimeout(30).GetAsync()
  • HttpClient 必须每次传 CancellationToken 或设实例级 Timeout(不推荐,因影响所有请求)

真正要注意的是:Flurl.Http 的 Timeout 包含 DNS 解析、连接、发送、接收全过程;而 HttpClient.Timeout 仅作用于整个请求周期,行为一致。但 Flurl 默认值更“保守”,上线前务必核对是否符合业务 SLA。

相关专题

更多
switch语句用法
switch语句用法

switch语句用法:1、Switch语句只能用于整数类型,枚举类型和String类型,不能用于浮点数类型和布尔类型;2、每个case语句后面必须跟着一个break语句,以防止执行其他case的代码块,没有break语句,将会继续执行下一个case的代码块;3、可以在一个case语句中匹配多个值,使用逗号分隔;4、Switch语句中的default代码块是可选的等等。

534

2023.09.21

Java switch的用法
Java switch的用法

Java中的switch语句用于根据不同的条件执行不同的代码块。想了解更多switch的相关内容,可以阅读本专题下面的文章。

415

2024.03.13

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

481

2023.08.10

default gateway怎么配置
default gateway怎么配置

配置default gateway的步骤:1、了解网络环境;2、获取路由器IP地址;3、登录路由器管理界面;4、找到并配置WAN口设置;5、配置默认网关;6、保存设置并退出;7、检查网络连接是否正常。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

220

2023.12.07

http500解决方法
http500解决方法

http500解决方法有检查服务器日志、检查代码错误、检查服务器配置、检查文件和目录权限、检查资源不足、更新软件版本、重启服务器或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

360

2023.11.09

http请求415错误怎么解决
http请求415错误怎么解决

解决方法:1、检查请求头中的Content-Type;2、检查请求体中的数据格式;3、使用适当的编码格式;4、使用适当的请求方法;5、检查服务器端的支持情况。更多http请求415错误怎么解决的相关内容,可以阅读下面的文章。

410

2023.11.14

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

3

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.4万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号