1.HttpClient类使用存在的问题

HttpClient类的使用所存在的问题,百度搜索的文章一大堆,好多都是单纯文字描述,让人感觉不太好理解,为了更好理解HttpClient使用存在的问题,下面让我们通过代码跟示例来描述。

using(var client = new HttpClient())

传统关闭连接方法如上述代码所示,但当使用using语句释放HttpClient对象的时候,套接字(socket)也不会立即释放,下面我们通过请求aspnetmonsters站点的示例来验证下:

class Program
{static void Main(string[] args){Console.WriteLine("Starting connections");var g = GetAsync();g.Wait();Console.WriteLine("Connections done");Console.ReadKey();}static async Task GetAsync(){for (int i = 0; i < 5; i++){using (var client = new HttpClient()){var result = await client.GetAsync("http://aspnetmonsters.com/");Console.WriteLine(result.StatusCode);}}}
}

输出结果:

控制台打印出五条请求站点返回状态的信息,下面我们通过netstat工具打印出五个请求连接套接字状态:

应用程序已经运行结束了(结束连接),但是打印结果显示连接状态仍然是TIME_WAIT,也就是说在此状态期间仍然在观察是否有数据包进入连接(如果连接等待中有任何数据包仍然会通过),因为它们可能在某个地方被网络延迟,这是我从tcpstate窃取的TCP / IP状态图。

Windows将在此状态下保持连接240秒(由其设置[HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\Tcpip\Parameters\TcpTimedWaitDelay])。Windows可以快速打开新套接字的速度有限,因此如果您耗尽连接池,那么您可能会看到如下错误:

而怎么做才可以减少套接字的浪费呢?我们在上述代码中把每次循环中创建的HttpClient对象拉到Main外定义为一个共享的静态实例:

class Program
{private static HttpClient client = new HttpClient();static void Main(string[] args){Console.WriteLine("Starting connections");var g = GetAsync();g.Wait();Console.WriteLine("Connections done");Console.ReadKey();}static async Task GetAsync(){for (int i = 0; i < 5; i++){var result = await client.GetAsync("http://aspnetmonsters.com/");Console.WriteLine(result.StatusCode);}}
}

应用程序运动完毕之后,我们再通过netstat工具打印出五个请求连接套接字状态,这时候会看到信息如下:

通过共享一个实例,减少了套接字的浪费,实际上由于套接字重用而传输快一点。
总结:
●在创建HttpClient实例的时候,最好是静态(static )实例。
●不要用using包装HttpClient对象。
在.NET Core 2.1版本之后引入的 HttpClientFactory解决了HttpClient的所有痛点。有了 HttpClientFactory,我们不需要关心如何创建HttpClient,又如何释放它。通过它可以创建具有特定业务的HttpClient,而且可以很友好的和 DI 容器结合使用,更为灵活。下面以 ASP.NET Core为例介绍HttpClientFactory的四种使用方式。

2.HttpClientFactory 的多种使用方式

可以通过多种使用方式在应用程序中使用HttpClientFactory。

2.1直接使用HttpClientFactory

在Startup.ConfigureServices方法中,通过在IServiceCollection上调用AddHttpClient扩展方法可以注册IHttpClientFactory服务。
services.AddHttpClient();
注册服务后,我们新建BasicUsageModel类使用IHttpClientFactory创建HttpClient实例:

public class BasicUsageModel
{private readonly IHttpClientFactory _clientFactory;public IEnumerable<GitHubBranch> Branches { get; private set; }public bool GetBranchesError { get; private set; }public BasicUsageModel(IHttpClientFactory clientFactory){_clientFactory = clientFactory;}public async Task OnGet(){var request = new HttpRequestMessage(HttpMethod.Get,"https://api.github.com/repos/aspnet/AspNetCore.Docs/branches");request.Headers.Add("Accept", "application/vnd.github.v3+json");request.Headers.Add("User-Agent", "HttpClientFactory-Sample");var client = _clientFactory.CreateClient();var response = await client.SendAsync(request);if (response.IsSuccessStatusCode){Branches = await response.Content.ReadAsAsync<IEnumerable<GitHubBranch>>();}else{GetBranchesError = true;Branches = Array.Empty<GitHubBranch>();}}
}
public class GitHubBranch
{public string name { get; set; }
}

以这种方式直接在使用IHttpClientFactory的类中调用CreateClient方法创建HttpClient实例。然后在Controller中调用BasicUsageModel类:

public class HomeController : Controller
{private readonly IHttpClientFactory _clientFactory;public HomeController(IHttpClientFactory clientFactory){_clientFactory = clientFactory;}public IActionResult Index(){BasicUsageModel model = new BasicUsageModel(_clientFactory);var task = model.OnGet();task.Wait();List<GitHubBranch> list = model.Branches.ToList();return View(list);}
}

2.2使用命名客户端

如果应用程序需要有许多不同的HttpClient用法(每种用法的服务配置都不同),可以视情况使用命名客户端。可以在HttpClient中注册时指定命名Startup.ConfigureServices的配置。

services.AddHttpClient("github", c =>
{c.BaseAddress = new Uri("https://api.github.com/");// Github API versioningc.DefaultRequestHeaders.Add("Accept", "application/vnd.github.v3+json");// Github requires a user-agentc.DefaultRequestHeaders.Add("User-Agent", "HttpClientFactory-Sample");
});

上面的代码调用AddHttpClient,同时提供名称“github”。此客户端应用了一些默认配置,也就是需要基址和两个标头来使用GitHub API。每次调用CreateClient时,都会创建HttpClient 的新实例,并调用配置操作。要使用命名客户端,可将字符串参数传递到CreateClient。指定要创建的客户端的名称:

public class NamedClientModel : PageModel
{private readonly IHttpClientFactory _clientFactory;public IEnumerable<GitHubPullRequest> PullRequests { get; private set; }public bool GetPullRequestsError { get; private set; }public bool HasPullRequests => PullRequests.Any();public NamedClientModel(IHttpClientFactory clientFactory){_clientFactory = clientFactory;}public async Task OnGet(){var request = new HttpRequestMessage(HttpMethod.Get,"repos/aspnet/AspNetCore.Docs/pulls");var client = _clientFactory.CreateClient("github");var response = await client.SendAsync(request);if (response.IsSuccessStatusCode){PullRequests = await response.Content.ReadAsAsync<IEnumerable<GitHubPullRequest>>();}else{GetPullRequestsError = true;PullRequests = Array.Empty<GitHubPullRequest>();}}
}
public class GitHubPullRequest
{public string url { get; set; }public int? id { get; set; }public string node_id { get; set; }
}

在上述代码中,请求不需要指定主机名。可以仅传递路径,因为采用了为客户端配置的基址。在Controller中调用方法如上个示例。

2.3使用类型化客户端

什么是“类型化客户端”?它只是DefaultHttpClientFactory注入时配置的HttpClient。
下图显示了如何将类型化客户端与HttpClientFactory结合使用:

类型化客户端提供与命名客户端一样的功能,不需要将字符串用作密钥。它们提供单个地址来配置特定HttpClient并与其进行交互。例如,单个类型化客户端可能用于单个后端终结点,并封装此终结点的所有处理逻辑。另一个优势是它们使用 DI 且可以被注入到应用中需要的位置。
类型化客户端在构造函数中接收HttpClient参数:

public class GitHubService
{public HttpClient Client { get; }public GitHubService(HttpClient client){client.BaseAddress = new Uri("https://api.github.com/");// GitHub API versioningclient.DefaultRequestHeaders.Add("Accept","application/vnd.github.v3+json");// GitHub requires a user-agentclient.DefaultRequestHeaders.Add("User-Agent","HttpClientFactory-Sample");Client = client;}public async Task<IEnumerable<GitHubIssue>> GetAspNetDocsIssues(){var response = await Client.GetAsync(
"/repos/aspnet/AspNetCore.Docs/issues?state=open&sort=created&direction=desc");response.EnsureSuccessStatusCode();var result = await response.Content.ReadAsAsync<IEnumerable<GitHubIssue>>();return result;}
}
public class GitHubIssue
{public string url { get; set; }public int? id { get; set; }public string node_id { get; set; }
}

在上述代码中,配置转移到了类型化客户端中。HttpClient对象公开为公共属性。可以定义公开HttpClient功能的特定于API的方法。GetAspNetDocsIssues方法从GitHub存储库封装查询和分析最新待解决问题所需的代码。
要注册类型化客户端,可在Startup.ConfigureServices中使用通用的AddHttpClient扩展方法,指定类型化客户端类:

services.AddHttpClient<GitHubService>();

使用DI将类型客户端注册为暂时客户端。可以直接插入或使用类型化客户端:

public class TypedClientModel : PageModel
{private readonly GitHubService _gitHubService;public IEnumerable<GitHubIssue> LatestIssues { get; private set; }public bool HasIssue => LatestIssues.Any();public bool GetIssuesError { get; private set; }public TypedClientModel(GitHubService gitHubService){_gitHubService = gitHubService;}public async Task OnGet(){try{LatestIssues = await _gitHubService.GetAspNetDocsIssues();}catch (HttpRequestException){GetIssuesError = true;LatestIssues = Array.Empty<GitHubIssue>();}}
}

参考文献:
在ASP.NET Core中使用IHttpClientFactory发出HTTP请求
你正在以错误方式使用 HttpClient,这将导致软件受损

转载于:https://www.cnblogs.com/wzk153/p/10945313.html

(6)ASP.NET Core 中使用IHttpClientFactory发出HTTP请求相关推荐

  1. ASP.NET Core 中使用IHttpClientFactory发出HTTP请求

    1.HttpClient类使用存在的问题 HttpClient类的使用所存在的问题,百度搜索的文章一大堆,好多都是单纯文字描述,让人感觉不太好理解,为了更好理解HttpClient使用存在的问题,下面 ...

  2. 如何在ASP.NET Core中使用SignalR构建与Angular通信的实时通信应用程序

    图片 假设我们要创建一个监视Web应用程序,该应用程序为用户提供了一个能够显示一系列信息的仪表板,这些信息会随着时间的推移而更新. 第一种方法是在定义的时间间隔(轮询)定期调用API 以更新仪表板上的 ...

  3. ASP.NET Core 中文文档 第三章 原理(13)管理应用程序状态

    原文:Managing Application State 作者:Steve Smith 翻译:姚阿勇(Dr.Yao) 校对:高嵩 在 ASP.NET Core 中,有多种途径可以对应用程序的状态进行 ...

  4. 如何在 ASP.NET Core 中使用 HttpClientFactory ?

    ASP.Net Core 是一个开源的,跨平台的,轻量级模块化框架,可用它来构建高性能的Web程序,这篇文章我们将会讨论如何在 ASP.Net Core 中使用 HttpClientFactory. ...

  5. ASP.NET Core中的Http缓存

    ASP.NET Core中的Http缓存 Http响应缓存可减少客户端或代理对 web服务器发出的请求数.响应缓存还减少了 web服务器生成响应所需的工作量.响应缓存由 Http请求中的 header ...

  6. ASP.NET Core中使用MediatR实现命令和中介者模式

    作者:依乐祝 原文地址:https://www.cnblogs.com/yilezhu/p/9866068.html 在本文中,我将解释命令模式,以及如何利用基于命令模式的第三方库来实现它们,以及如何 ...

  7. ASP.NET Core 中的管道机制

    首先,很感谢在上篇文章 C# 管道式编程 中给我有小额捐助和点赞的朋友们,感谢你们的支持与肯定.希望我的每一次分享都能让彼此获得一些收获,当然如果我有些地方叙述的不正确或不当,还请不客气的指出.好了, ...

  8. 在 ASP.NET Core 中安装 MVC

    目录 本文出自<从零开始学 ASP.NET CORE MVC>目录  视频课程效果更佳:从零开始学 Asp.Net Core MVC 在ASP.NET Core 中安装 MVC 在本视频中 ...

  9. 如何在ASP.NET Core中使用Azure Service Bus Queue

    原文:USING AZURE SERVICE BUS QUEUES WITH ASP.NET CORE SERVICES 作者:damienbod[1] 译文:如何在ASP.NET Core中使用Az ...

  10. ASP.NET Core中Ocelot的使用:基于服务发现的负载均衡

    本系列相关文章: <ASP.NET Core中Ocelot的使用:API网关的应用> <ASP.NET Core中Ocelot的使用:基于Spring Clound Netflix ...

最新文章

  1. Go Code Review Comments 翻译 编写优雅golang代码
  2. NoSQL(1)之 Redis的五大数据类型使用方法的详细介绍
  3. django 表单html5,我们如何在django管理表单中添加动态html5数据属性
  4. java 上传文件到服务器_java上传文件到OSS云服务器(二)
  5. 机器学习之数据转换(七)——降维
  6. A/B Testing
  7. python invalid character_python提示invalid character in identifier
  8. 外网登录homeassistant
  9. 给大家爆个秘密:怎样迅速提高你的校内人人网人气
  10. 查看链接文件的最终目标的多种方法
  11. 目标检测经典论文——YOLOv1论文翻译(纯中文版):YOLO:统一的实时目标检测
  12. AUTOCAD——文件管理
  13. spark on k8s:apache YuniKorn(Incubating)的助力
  14. Python实现1~100猜数字小游戏
  15. echarts修改标题字体大小、颜色、位置、内容
  16. STM8S之STVD问题解决47 can't openfile crtsi0.sm8
  17. iOS开发的Sketch之旅
  18. 绩效考核过程中使用系统的必要性
  19. ChemDraw 2D与ChemBio 3D之间的信息转换
  20. 阿飞学习mina框架

热门文章

  1. 连接Excel时出现未指定的错误
  2. OsmocomBB编译及GSM嗅探问题(转)
  3. VS 2010 安装 .net framework2.0/3.0/3.5
  4. SqlServer语句重命名数据库名称
  5. QA:阿里云K8s启动容器后无法访问网络
  6. libevent源码深度剖析-张亮
  7. java 字符串转long_如何在Java中将String转换为long?
  8. 网络摄像机 c++ 抓拍_IP摄像机和工业摄像机怎么区分 IP摄像机和工业摄像机差异...
  9. 创建zookeeper客户端
  10. vagrant虚拟机网络设置