当前位置: 首页 > article >正文

C# 使用HttpClient进行Post请求总是出现超时问题的优化

最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求。
在使用apipost发现并发10个5分钟也没有问题,那么问题就出在我的请求端了。

优化结论

我直接上优化结论吧,就不放上老的代码了。需要从以下几个点来优化。

单例HttpClient

问题:如果 HttpClient 实例频繁创建和销毁,可能导致连接池中的资源被占满,新的请求需要等待释放资源,从而造成长时间的延迟。

首先单例HttpClient,每次请求都会创建一个新的 HttpClient 实例。HttpClient 的短生命周期会导致以下问题:
1,频繁建立和销毁连接,无法复用已有的连接池。
2,增加连接开销,可能导致长时间等待(尤其在并发请求时)。
所以我们直接

private static readonly HttpClient client = new HttpClient
{
    Timeout = TimeSpan.FromSeconds(15) // 设置超时时间
};

连接池耗尽和并发

合理设置 ServicePointManager.DefaultConnectionLimit,因为就算是单例的HttpClient也会有连接数的限制。我们看看这个参数说明:

// 摘要:
//     Gets or sets the maximum number of concurrent connections allowed by a System.Net.ServicePoint
//     object.
//
// 返回结果:
//     The maximum number of concurrent connections allowed by a System.Net.ServicePoint
//     object. The default connection limit is 10 for ASP.NET hosted applications and
//     2 for all others. When an app is running as an ASP.NET host, it is not possible
//     to alter the value of this property through the config file if the autoConfig
//     property is set to true. However, you can change the value programmatically when
//     the autoConfig property is true. Set your preferred value once, when the AppDomain
//     loads.
//
// 异常:
//   T:System.ArgumentOutOfRangeException:
//     System.Net.ServicePointManager.DefaultConnectionLimit is less than or equal to
//     0.

有一句是重点
ASP的默认连接限制是10。. NET托管应用程序和其他的都是2。
我可能有时又3-4个并发,可能问题在这里,那么我直接设置100个就足够满足我的程序了。

ServicePointManager.DefaultConnectionLimit = 100; // 调高默认连接限制

并发异步

如果你的程序有很高的并发,可能会耗尽你的CPU,那么需要使用异步。

HttpResponseMessage response = await client.PostAsync(url, content);

最终优化后

我最终的代码状态如下:

public async Task<string> PostFormResult(string url, string parm)
{
    Log("PostFormResult 开始请求: " + url + ", parm: " + parm);
    try
    {
        byte[] buf = Encoding.UTF8.GetBytes(parm);
        using (HttpContent content = new ByteArrayContent(buf))
        {
        	//这里我是表单,可以换成json
            content.Headers.ContentType = new System.Net.Http.Headers.MediaTypeHeaderValue("application/x-www-form-urlencoded");
			//content.Headers.ContentType = new System.Net.Http.Headers.MediaTypeHeaderValue("application/json");
			//添加Token	
		    //client.DefaultRequestHeaders.Add("Authorization", "Bearer " + token);

            HttpResponseMessage res = await client.PostAsync(url, content);

            if (res.IsSuccessStatusCode)
            {
                string json = await res.Content.ReadAsStringAsync();
                Log("PostFormResult请求成功: " + json);
                return json;
            }
            else
            {
                Warning("PostFormResult请求失败: " + res.StatusCode);
            }
        }

    }
    catch (HttpRequestException ex)
	{
	    Warning("请求Post出现错误: " + ex.Message);
	}
	catch (Exception ex)
	{
	    Warning($"请求Post出现错误: {ex.Message}");
	}
    return string.Empty;
}

我的请求会同时出现了4个。所以超过了并发所以产生了问题,修改后就没有问题了。


http://www.kler.cn/a/517108.html

相关文章:

  • python实现http文件服务器访问下载
  • 每天五分钟深度学习pytorch:基于VGG神经网络完成CAFIR10的识别
  • 激光雷达和相机早期融合
  • 双指针+前缀和习题(一步步讲解)
  • 优化使用 Flask 构建视频转 GIF 工具
  • 全球化趋势与中资企业出海背景
  • 一文了解二叉树的基本概念
  • AD7606, 逐次逼近型ADC以及一次被GPT坑了的过程.
  • vue + element-ui 组件样式缺失导致没有效果
  • Go中的三种锁
  • 实践深度学习:构建一个简单的图像分类器
  • c语言中的位域详解
  • mac 通过 Homebrew 安装 git 遇到的问题
  • ECS中实现Nginx四层和七层负载均衡以及ALB/NLB实现负载均衡
  • react install
  • Langchain+文心一言调用
  • SOME/IP服务接口
  • 干货分享|算法竞赛真题讲解2
  • Liunx上Jenkins 持续集成 Java + Maven + TestNG + Allure + Rest-Assured 接口自动化项目
  • 从语音识别到图像识别:AI如何“看”和“听”
  • 状态模式——C++实现
  • 分布式 IO 模块携手 PLC,开启设备车间降本增效新篇章
  • git cherry-pick从一个分支中选择一个或多个提交(commit)并将其应用到当前分支
  • OpenStack基础架构
  • 以Python 做服务器,N Robot 做客户端,小小UI,拿捏
  • 如何使用Midjourney生成中国蛇年的灵蛇绘画作品