返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C#中HttpClient使用注意(预热与长连接)
  • 543
分享到

C#中HttpClient使用注意(预热与长连接)

2024-04-02 19:04:59 543人浏览 八月长安
摘要

最近在测试一个第三方api,准备集成在我们的网站应用中。API的调用使用的是.net中的HttpClient,由于这个API会在关键业务中用到,对调用API的整体响应速度有严格要求,

最近在测试一个第三方api,准备集成在我们的网站应用中。API的调用使用的是.net中的HttpClient,由于这个API会在关键业务中用到,对调用API的整体响应速度有严格要求,所以对HttpClient有了格外的关注。

开始测试的时候,只在客户端通过HttpClient用PostAsync发了一个http post请求。测试时发现,从创建HttpClient实例,到发出请求,到读取到服务器的响应数据总耗时在2s左右,而且多次测试都是这样。2s的响应速度当然是无法让人接受的,我们希望至少控制在100ms以内。于是开始追查这个问题的原因。

在API的返回数据中包含了该请求在服务端执行的耗时,这个耗时都在20ms以内,问题与服务端API无关。于是把怀疑点放到了网络延迟上,但ping服务器的响应时间都在10ms左右,网络延迟的可能性也不大。

当我们正准备换一个网络环境进行测试时,突然想到,我们的测试方式有些问题。我们只通过HttpClient发了一个PostAsync请求,假如HttpClient在第一次调用时存在某种预热机制(比如在EF中就有这样的机制),现在2s的总耗时可能大多消耗在HttpClient的预热上。

于是修改测试代码,将调用由1次改为100次,然后恍然大悟地发现——只有第1次是2s,接下来的99次都在100ms以内。果然是HttpClient的某种预热机制在搞鬼!

既然知道了是HttpClient预热机制的原因,那我们可以帮HttpClient进行热身,减少第一次请求的耗时。我们尝试了一种预热方式,在正式发http post请求之前,先发一个http head请求,代码如下:

_httpClient.SendAsync(new HttpRequestMessage {
                    Method = new HttpMethod("HEAD"), 
                    RequestUri = new Uri(BASE_ADDRESS + "/") })
                .Result.EnsureSuccessStatusCode();

经测试,通过这种热身方法,可以将第一次请求的耗时由2s左右降到1s以内(测试结果是700多ms)。

在知道第1次HttpClient请求耗时2s的真相之后,我们将目光转向了剩下的99次耗时100ms以内的请求,发现绝大部分请求都在50ms以上。有没有可能将之降至50ms以下?而且,之前一直有这样的纠结:每次调用是不是一定要对HttpClient进行Dispose()?是不是要将HttpClient单例或者静态化(声明为静态变量)?借此机会一起研究一下。

在HttpClient的背后,有一个对请求响应速度有着不容忽视影响的东东——tcp连接。一个HttpClient实例会关联一个TCP连接,在对HttpClient进行Dispose时,会关闭TCP连接(我们用Wireshark进行网络抓包也验证了这一点)。

在之前的测试中,我们每次用HttpClient发请求时,都是新建一个HttpClient实例,用完就对它进行Dispose,代码如下:

using (var httpClient = new HttpClient() { BaseAddress = new Uri(BASE_ADDRESS) })
{
    httpClient.PostAsync("/", new FORMUrlEncodedContent(parameters));
}

所以每次请求时都要经历新建TCP连接->传数据->关闭连接(也就是通常所说的短连接),而且雪上加霜的是请求用的是https,建立TCP连接时还需要一个基于公私钥加解密的key exchange过程:Client Hello -> Server Hello -> Certificate -> Client Key Exchange -> New Session Ticket。

如果我们想将请求响应时间降至50ms以下,就必须从这个地方下手——重用TCP连接(也就是通常所说的长连接)。要实现长连接,首先需要的就是在HttpClient第1次请求后不关闭TCP连接(不调用Dispose方法);而要让后续的请求继续使用这个未关闭的TCP连接,我们必须要使用同一个HttpClient实例;而要使用同一个HttpClient实例,就得实现HttpClient的单例或者静态化。之前的3 个问题,由于要解决第1个问题,后2个问题变成了别无选择。

为了实现长连接,我们将HttpClient的调用代码改为如下的样子:

然后测试一下请求响应时间:

  Elapsed:750ms
  Elapsed:31ms
  Elapsed:30ms
  Elapsed:43ms
  Elapsed:27ms
  Elapsed:29ms
  Elapsed:28ms
  Elapsed:35ms
  Elapsed:36ms
  Elapsed:31ms
  ....

除了第1次请求,接下来的99次请求绝大多数都在50ms以内。TCP长连接的效果必须的!

通过Wireshak抓包也验证了长连接的效果:

Wireshak抓包

这时,你

public class HttpClientTest
{ 
    private static readonly HttpClient _httpClient;

    static HttpClientTest()
    {
        _httpClient = new HttpClient() { BaseAddress = new Uri(BASE_ADDRESS) };

        //帮HttpClient热身
        _httpClient.SendAsync(new HttpRequestMessage {
                Method = new HttpMethod("HEAD"), 
                RequestUri = new Uri(BASE_ADDRESS + "/") })
            .Result.EnsureSuccessStatusCode();
    }

    public async Task<string> PostAsync()
    {
        var response = await _httpClient.PostAsync("/", new FormUrlEncodedContent(parameters));

        return await response.Content.ReadAsStringAsync();
    }
}

也许会产生这样的疑问:将HttpClient声明为静态变量,会不会存在线程安全问题?我们当时也有这样的疑问,后来在stackoverflow上找到了答案:

As per the comments below (thanks @ischell), the following instance methods are thread safe (all async):
CancelPendingRequests
DeleteAsync
GetAsync
GetByteArrayAsync
GetStreamAsync
GetStringAsync
PostAsync
PutAsync
SendAsync

HttpClient的所有异步方法都是线程安全的,放心使用。

到这里,HttpClient的问题是不是可以完美收官了?。。。稍等,还有一个问题。

客户端虽然保持着TCP连接,但TCP连接是两口子的事,服务器端呢?你不告诉服务器,服务器怎么知道你要一直保持TCP连接呢?对于客户端,保持TCP连接的开销不大;但是对于服务器,则完全不一样的,如果默认都保持TCP连接,那可是要保持成千上万客户端的连接啊。所以,一般的WEB服务器都会根据客户端的诉求来决定是否保持TCP连接,这就是keep-alive存在的理由。

所以,我们还要给HttpClient增加一个Connection:keep-alive的请求头,代码如下:

_httpClient.DefaultRequestHeaders.Connection.Add("keep-alive");

现在终于可以收官了。但是肯定不完美,分享的只是解决问题的过程。

到此这篇关于C#中HttpClient使用注意(预热与长连接)的文章就介绍到这了,更多相关C# HttpClient内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C#中HttpClient使用注意(预热与长连接)

本文链接: https://lsjlt.com/news/139062.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C#中HttpClient使用注意(预热与长连接)
    最近在测试一个第三方API,准备集成在我们的网站应用中。API的调用使用的是.NET中的HttpClient,由于这个API会在关键业务中用到,对调用API的整体响应速度有严格要求,...
    99+
    2024-04-02
  • 使用HTTPclient保持长连接
    目录HTTPclient保持长连接首先解释一下什么是长连接如何在java中实现一个长连接呢httpclient因为保持永久长连接造成连接吊死的问题添加策略后,问题解决HTTPclie...
    99+
    2024-04-02
  • C#中HttpClient使用注意事项有哪些
    小编给大家分享一下C#中HttpClient使用注意事项有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!最近在测试一个第三方API,准备集成在我们的网站应用中...
    99+
    2023-06-29
  • C++预处理连接方法怎么使用
    这篇文章主要讲解了“C++预处理连接方法怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++预处理连接方法怎么使用”吧!C++预处理连接(Preprocessor Concatena...
    99+
    2023-07-05
  • Java中使用JDBC连接数据库的示例代码与注意事项
    本篇内容主要讲解“Java中使用JDBC连接数据库的示例代码与注意事项”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java中使用JDBC连接数据库的示例代码与注意事项”吧!实例代码如下:&nb...
    99+
    2023-06-17
  • 【C/C++与MySql的连接及使用详解】
    文章目录 概要:本期主要讲解C或者C++怎么在程序中连接MySql数据库。主要步骤一、MySql的链接库二、项目环境配置三、项目内调用MySql类进行数据库操作1.准备工作2.连接数据库3.数据库建表4.向表中插入数据5.查询表中数...
    99+
    2023-08-20
    mysql c语言 c++
  • 使用idea连接mysql要注意哪些事项
    使用IDEA连接MySQL时需要注意以下事项:1. 确保已安装MySQL服务器,并已在服务器上创建了要连接的数据库。2. 在IDEA...
    99+
    2023-10-26
    idea 数据库
  • C#中使用CLR需要注意什么
    这篇文章主要为大家展示了“C#中使用CLR需要注意什么”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“C#中使用CLR需要注意什么”这篇文章吧。1、C# CLR之foreach的性能问题 fore...
    99+
    2023-06-17
  • 使用HikariCP连接池常用配置讲解及注意事项
    使用HikariCP连接池常用配置讲解及注意事项 常遇到的几种错误Possibly consider using a shorter maxLifetime valueConnection i...
    99+
    2023-09-15
    mysql 数据库 java spring
  • c++中使用mfc框架要注意什么
    在使用MFC框架开发C++应用程序时,需要注意以下几点: MFC类的命名规范:MFC中的类名前缀通常是C(例如CDialog、C...
    99+
    2024-02-29
    c++ mfc
  • C语言中#pragma pack(1)的用法与注意点
    目录一:何时使用二. 为什么使用#pragma pack(1)三.注意点四.#pragma pack()的一些用法五.题目附:C语言慎用#pragma pack(1)命令总结一:何时...
    99+
    2023-02-21
    c语言 #pragma pack(1) c语言中pragma怎么用 pragma使用例子pack
  • C语言指针的使用技巧与注意事项
    C语言指针的应用技巧与注意事项 一、引言 作为一种面向过程的编程语言,C语言具有高效、灵活的特点。而指针则是C语言中一个非常重要的概念,对于理解和掌握C语言来说至关重要。本文将介绍C语...
    99+
    2024-02-26
    传递 动态内存 野指针 注意事项:空指针
  • C/C++开发中extern的一些使用注意事项
    目录前言数组与指针的区别具体分析extern "C"前言 前些日子,有友友问了我这样的一道问题: 数组通过外部声明为指针时,数组和指针是不能互换使用的;那么请思...
    99+
    2023-01-04
    C/C++开发extern使用事项 C C++ extern
  • 微信小程序中怎么使用WebSocket实现长连接
    这篇文章主要讲解了“微信小程序中怎么使用WebSocket实现长连接”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“微信小程序中怎么使用WebSocket实现长连接”吧!项目使用的技术栈数据请...
    99+
    2023-06-26
  • C语言 volatile与const同时使用应注意的问题
    const和volatile放在一起的意义在于: (1)本程序段中不能对a作修改,任何修改都是非法的,或者至少是粗心,编译器应该报错,防止这种粗心;(2)另一个程序段则完全有可能修改...
    99+
    2022-11-15
    C语言 volatile const
  • C#中怎么使用OleDbConnection连接读取Excel
    在C#中使用OleDbConnection连接读取Excel文件,可以按照以下步骤进行操作:1. 引入System.Data.Ole...
    99+
    2023-08-09
    C# Excel
  • 使用JDBC连接Mysql数据库时需要注意哪些事项
    使用JDBC连接Mysql数据库时需要注意哪些事项?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。首先理清几个概念:JDBC:j...
    99+
    2024-04-02
  • C语言中使用break要注意哪些事项
    在C语言中,使用break语句时需要注意以下几点事项:1. break只能用于循环语句和switch语句中,用来跳出当前循环或swi...
    99+
    2023-10-12
    C语言
  • Java中Process类的使用与注意事项说明
    目录Process类的使用与注意事项说明1、在项目开发中2、在这里就需要认识一下process类3、来说说今天业务需求[waitfor()]:4、前不久遇到一个奇怪的问题就是ajax...
    99+
    2024-04-02
  • 使用C#中的String.Concat函数连接字符串
    标题:使用C#中的String.Concat函数连接字符串在C#编程中,字符串是一种常见的数据类型,我们经常需要对字符串进行连接操作。C#提供了多种方式来实现字符串的连接,其中一种常用的方法是使用String.Concat函数。本文将介绍S...
    99+
    2023-11-18
    string concat C#中的关键词:C#
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作