游乐游手机版
首页/编程语言/文章详情

C#怎么发送HTTP请求 C#如何使用HttpClient调用REST API接口获取数据【网络】

时间:2026-05-05 11:50
C 如何发送HTTP请求?详解HttpClient调用REST API接口获取数据的最佳实践【网络编程】 HttpClient 必须复用,切忌每次新建实例 许多C 开发者初次使用HttpClient时,常犯的一个错误是在方法内部直接 new HttpClient()。这种做法看似便捷,实则隐患巨大。

C#如何发送HTTP请求?详解HttpClient调用REST API接口获取数据的最佳实践【网络编程】

C#怎么发送HTTP请求 C#如何使用HttpClient调用REST API接口获取数据【网络】

HttpClient 必须复用,切忌每次新建实例

许多C#开发者初次使用HttpClient时,常犯的一个错误是在方法内部直接 new HttpClient()。这种做法看似便捷,实则隐患巨大。频繁创建HttpClient实例会导致底层TCP套接字端口无法及时释放,最终可能引发端口耗尽和 SocketException 异常。此外,每次新建连接都无法复用DNS解析缓存和连接池,严重拖累应用程序性能。

那么,如何正确复用HttpClient呢?在 .NET Core 2.1 及更高版本中,最佳方案是通过依赖注入,使用 IServiceCollection.AddHttpClient() 将其注册为单例或作用域服务。对于控制台应用或简单脚本,一个有效的方法是将其声明为静态只读字段。

  • 典型的错误示例:在方法内部使用 var client = new HttpClient(); —— 这会导致每次调用都创建新实例。
  • 推荐的静态复用方案private static readonly HttpClient _client = new HttpClient();
  • 重要提醒:除非你完全确定该实例的生命周期已结束且后续不再使用,否则不要手动调用 _client.Dispose(),应交由框架管理其生命周期。

GET请求参数编码:务必使用Uri.EscapeDataString或QueryHelpers

构建GET请求的查询字符串是另一个高频错误点。直接进行字符串拼接,例如手动编写 ?key=value&key2=value2,当参数值包含空格、中文、&、=等特殊字符时,会破坏URL结构,导致服务端返回400错误或解析失败。特别是当参数来自用户输入或数据库时,风险极高。

因此,务必避免手动拼接。推荐使用 UriBuilder 配合 QueryHelpers(需引用 Microsoft.AspNetCore.WebUtilities NuGet包)来安全构建URL。

  • 推荐的安全构建方式
    var uri = new UriBuilder("https://api.example.com/users");
    uri.Query = QueryHelpers.AddQueryString("", new Dictionary {
        ["name"] = "张三",
        ["page"] = "1"
    });
    var response = await _client.GetAsync(uri.ToString());
  • 不安全的做法:使用字符串插值如 $"https://...?name={name}"。这会导致中文字符乱码,且参数中的 & 符号可能截断查询字符串,引发未知问题。

POST JSON数据:必须设置Content-Type并使用StringContent

发送POST请求提交JSON数据时,常见的疏忽是忘记设置正确的请求头。许多新手会这样写:client.PostAsync(url, new StringContent(json, Encoding.UTF8))。虽然请求体是JSON文本,但缺少 Content-Type: application/json 头信息,服务端无法正确解析,通常返回415 Unsupported Media Type错误。

  • 正确的完整代码示例
    var json = JsonSerializer.Serialize(new { username = "alice", pwd = "123" });
    var content = new StringContent(json, Encoding.UTF8, "application/json");
    var response = await _client.PostAsync("https://api.example.com/login", content);
  • 需要警惕的混淆:切勿使用 FormUrlEncodedContent 发送JSON。该类专为 application/x-www-form-urlencoded 格式设计,会将整个JSON字符串当作一个字段值编码,导致服务端无法解析出预期结构。
  • 补充说明:若调用的API需要Bearer Token认证,记得在发送请求前设置请求头:_client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", token);

异步方法必须使用await,避免.Result或.Wait()

在异步编程中,阻塞异步任务是严重错误。尤其在ASP.NET Core这类拥有同步上下文(SynchronizationContext)的Web应用环境中,在控制器中调用 task.Result.Wait() 极易导致线程死锁,使应用无响应。HttpClient 的核心方法如 GetAsyncPostAsync 均为真正的异步操作,必须使用 await 关键字。

  • 绝对要避免的写法var res = client.GetAsync(url).Result; 或使用 .Wait()
  • 在ASP.NET Core控制器中的正确做法:将Action方法标记为 async Task,并在方法体内使用 await
  • 在控制台程序中的处理:对于C# 7.1及以上版本,可使用 static async Task Main(string[] args) 作为入口点。若需在其他地方同步等待,可考虑使用 await MainAsync().ConfigureAwait(false) 来避免潜在的上下文问题。

总而言之,使用C#的HttpClient发送HTTP请求,真正的挑战往往不在于“如何发出请求”,而在于背后的连接复用策略、参数编码的安全性以及异步流程的妥善控制。这三个核心要点若未处理得当,接口在测试阶段可能表现正常,一旦上线,间歇性失败、响应缓慢乃至内存消耗持续上涨等问题便会接踵而至。这正是高效、稳定调用REST API接口的关键所在。

来源:https://www.php.cn/faq/2334256.html
上一篇c#如何使用switch语句_c#switch语句从入门到精通教程 下一篇如何从 Go 语言的空接口中安全提取 JSON 解析后的值
本站内容用于信息整理与展示,如有侵权或内容问题请及时联系处理。

相关推荐

补充同频道和同主题内容,方便继续浏览更多相关内容。

同类最新

继续查看同栏目最近更新的文章。

更多
CentOS与Golang打包常见兼容性问题探讨
编程语言 · 2026-07-01

CentOS与Golang打包常见兼容性问题探讨

CentOS与Golang打包的兼容性问题集中在glibc版本不匹配、交叉编译环境变量错误、依赖库缺失及Go依赖管理不规范。可通过Docker容器编译、选择兼容Go版本、正确设置GOOS GOARCH环境变量、安装对应开发包及使用GoModules解决。

CentOS中Fortran与Python如何协同工作从入门到实战完整教程
编程语言 · 2026-07-01

CentOS中Fortran与Python如何协同工作从入门到实战完整教程

在CentOS中,Fortran与Python可通过f2py、SWIG、共享库调用或subprocess协同。f2py封装Fortran为Python模块,支持数组运算;共享库需手动对齐数据类型;系统调用适合独立计算。

CentOS中Golang打包优化方法
编程语言 · 2026-07-01

CentOS中Golang打包优化方法

在CentOS中优化Golang编译打包,可显著提升编译速度并减小二进制文件体积。关键技巧包括:设置环境变量、使用Go模块管理依赖、编译时添加-ldflags= "-s-w "去除调试信息、利用UPX工具压缩、运行strip清理符号表,以及优化cgo内C代码的编译选项。综合运用这些方法能有效优化最终程序。

在CentOS系统中cpustat与其他工具协同使用的完整方法
编程语言 · 2026-07-01

在CentOS系统中cpustat与其他工具协同使用的完整方法

cpustat作为sysstat包的CPU监控工具,可通过管道与grep等命令配合过滤数据,利用脚本自动记录带时间戳的日志,或结合图形工具查看,也可格式化输出后接入Zabbix、Grafana等Web监控系统,实现可视化与告警。

CentOS中readdir与其他Linux发行版的差异
编程语言 · 2026-07-01

CentOS中readdir与其他Linux发行版的差异

CentOS基于RHEL,与Ubuntu、Debian、Fedora在包管理器(yum dnfvsapt)、默认文件系统(XFSvsext4)等存在差异,但readdir等系统调用遵循POSIX标准,行为一致。