在.NET Core使用 HttpClient 的正確方式

来源:https://www.cnblogs.com/xbhp/archive/2023/04/12/17309933.html
-Advertisement-
Play Games

前言 HttpClient 是 .NET Framework、.NET Core 或 .NET 5以上版本中的一個類,用於向 Web API 發送 HTTP 請求並接收響應。它提供了一些簡單易用的方法,如 GET、POST、PUT 和 DELETE,可以很容易地構造和發送 HTTP 請求,並處理響應 ...


前言

HttpClient 是 .NET Framework、.NET Core 或 .NET 5以上版本中的一個類,用於向 Web API 發送 HTTP 請求並接收響應。它提供了一些簡單易用的方法,如 GET、POST、PUT 和 DELETE,可以很容易地構造和發送 HTTP 請求,並處理響應數據。它是我們比較常用的官方HTTP請求組件,那麼你們都正確使用了嗎?本文將探討HttpClient的正確使用。

環境準備

首先我們用vs2022創建一個帶預設 WeatherForcast 模板的 Web API 應用程式,以及一個普通的API的程式,項目使用的是.NET6。

項目結構如下

圖片

兩個項目的功能點:

HttpClientTest - 返回天氣預報的Web API

HttpClientTest2 -這個項目將用HttpClient來請求HttpClientTest 的天氣預備。

接下來我們用4種方法來說明HttpClient的正確使用方法。

方法1

我們首先在HttpClientTest2 創建HttpClientTestController類,並寫一個請求天氣預備的方法,代碼如下:

namespace HttpClientTest2.Controllers
{
    [Route("api/[controller]")]
    [ApiController]
    public class HttpClientTestController : ControllerBase
    {
        [HttpGet]
        public async Task<string> TestHttpClient()
        {
            var url = "https://localhost:7281/WeatherForecast";
            #region 版本1
            var httpClient = new HttpClient();
            var response = await httpClient.GetAsync(url);
            return await response.Content.ReadAsStringAsync();
            #endregion
          }
       }
   }    

代碼寫完後,我們設置多項目啟動,讓這兩個項目同時啟動。

圖片

項目啟動後,執行項目HttpClientTest2 的TestHttpClient請求介面。多執行幾次。主要看看HttpClient後臺的執行情況。這裡可以用netstat來檢查http的請求情況。

打開一個CMD控制台程式。輸入如下代碼:

netstat -na | find "7281"

7281埠是我們請求站點HttpClientTest。多次點擊的效果如下:

圖片

由上面可以看出有多個請求,說明請求未關閉。接下來換第二種方法。

方法2

使用using命令來實現請求結束關閉請求,代碼如下:

   #region 版本2
    using (var httpClient = new HttpClient())
   {
        var response = await httpClient.GetAsync(url);
        return await response.Content.ReadAsStringAsync();
   }
   #endregion
   //歡迎公眾號:DOTNET開發跳槽

同樣我們多次請求,結果如下:

圖片

在這裡可以看到狀態“TIME_WAIT”,說明鏈接已經關閉,但實際情況鏈接還是占用著埠,在資源耗盡才會釋放。這就是套連接的問題,套接字耗儘是指伺服器上的可用套接字資源已經全部被占用,無法為新的連接提供服務。在 TCP/IP 網路通信中,每個埠上最多只能建立一個連接,這就限制了伺服器可以處理的連接數。當伺服器負載過高時,就可能導致套接字資源緊張,進而引發套接字耗盡問題。針對上面問題,繼續對HttpClient 改進。

方法3

這裡我們使用單例模式試一試。代碼如下:

public class HttpClientTestController : ControllerBase
    {
        private static HttpClient _httpClient;
        static HttpClientTestController()
        {
            _httpClient = new HttpClient();
        }
        //註意:有許多方法可以實現單例模式。在這裡使用了靜態實例方法。
        [HttpGet]
        public async Task<string> TestHttpClient()
        {
            var url = "https://localhost:7281/WeatherForecast";
            #region 版本3
            //var response = await _httpClient.GetAsync(url);
            //return await response.Content.ReadAsStringAsync();
            #endregion
        }
    }

代碼編寫完成後我們再試一試,結果如下:

圖片

因為使用了單例模式,沒有創建新實例使用了相同的連接。這種方法解決了套接字耗盡問題。但是,我們註意到有一個狀態為“已建立”的開放連接。如果有DNS更改或與網路相關的更改可能會影響連接,應用程式可能會失敗,需要重新啟動應用程式才能解決。這個方法也不是最理想的。

方法4

HttpClient是.NET內置方法,這裡可以通過使用 IHttpClientFactory 介面來實現,從而避免上面的問題。代碼如下:

public class HttpClientTestController : ControllerBase
    {
        private readonly IHttpClientFactory _httpClientFactory;
        public HttpClientTestController(IHttpClientFactory httpClientFactory)
        {
            _httpClientFactory = httpClientFactory;
        }
        [HttpGet]
        public async Task<string> TestHttpClient()
        {
            var url = "https://localhost:7281/WeatherForecast";
            #region 版本4
            var httpClient = _httpClientFactory.CreateClient();
            var response = await httpClient.GetAsync(url);
            return await response.Content.ReadAsStringAsync();
            #endregion
        }
//歡迎公眾號:DOTNET開發跳槽

使用IHttpClientFactory 的話,需要在Program.cs 中註入,代碼如下:

builder.Services.AddHttpClient();

同樣多次請求,然後執行netstat命令。效果如下:

圖片

從請求的狀態來看,通過使用 _httpClientFactory.CreateClient() 完美解決問題。

結語

本文用四種方法漸進講述了HttpClient的使用方法以及在使用過程中的問題,最終用IHttpClientFactory解決了出現的問題。希望本文對你有所收穫,歡迎留言或者吐槽。

源碼地址:https://github.com/xbhp/webapitest

參考:微軟官方文檔

來源公眾號:DotNet開發跳槽

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 說明 使用 VLD 記憶體泄漏檢測工具輔助開發時整理的學習筆記。本篇介紹如何在 Release 模式下使用 VLD。同系列文章目錄可見 《記憶體泄漏檢測工具》目錄 1. 思路概述 要在 RELEASE 模式下使用 VLD,必須在包含頭文件 vld.h 前預先定義 VLD_FORCE_ENABLE 巨集(參 ...
  • 拋硬幣實驗 random 模塊 import random random.randint(a, b) 返回一個隨機整數 N,範圍是:a <= N <= b random.choice("ilovefishc") 從 "ilovefishc" 這個字元串中隨機選出一個字元。 編寫一個雙色球的開獎模擬程 ...
  • 結構體 ​ 結構體是一種聚合的數據類型,是由零個或多個任意類型的值聚合成的實體。每個值稱為結構體的成員 結構體聲明 type + 結構體名 + struct + {成員列表} ​ :star:如果結構體成員名字是以大寫字母開頭的,那麼該成員就是導出的。這是Go語言導出規則決 定的。一個結構體可能同時 ...
  • PostgreSQL 備忘清單 PostgreSQL是一種特性非常齊全的自由軟體的對象-關係型資料庫管理系統(ORDBMS),是以加州大學電腦系開發的POSTGRES,4.2版本為基礎的對象關係型資料庫管理系統。POSTGRES的許多領先概念只是在比較遲的時候才出現在商業網站資料庫中。Postgr ...
  • 一、前提 pId需要傳入用來確認第一級的父節點,而且pId可以為null。 樹實體類必須實現:TreeNode介面 MyTreeVo必須有這三個屬性:id、pId、children 可以根據不同需求,配置TreeNode和MyTreeVo中固定的屬性 二、代碼 定義TreeNode介面 public ...
  • 使用java 11添加的HttpClient新API發送Http(Https)請求 HTTP客戶端是在Java 11中添加的。它可以用於通過網路請求HTTP資源。它支持 HTTP / 1.1和HTTP / 2(同步和非同步編程模型),將請求和響應主體作為反應流處理,並遵循熟悉的構建器模式。 參考文章: ...
  • 引入相關的依賴 <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.11</version> <scope>test</scope> </dependency> ​ <!--引⼊springb ...
  • 我最近幾次被問到關於 ExecutionContext 和 SynchronizationContext 的各種問題,例如它們之間的區別是什麼,“傳播”它們意味著什麼,以及它們與 C# 和 Visual Basic 中新的 async/await 關鍵字的關係。我想我會嘗試在這裡解決其中的一些問題。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...