如何利用HttpClientFactory實(shí)現(xiàn)簡(jiǎn)單的熔斷降級(jí)
前言
在2.1之后,有不少新東西,其中HttpClientFactory算是一個(gè)。HttpClientFactory涉及的東西也不算少,三四種clients , 請(qǐng)求中間件,與Polly的結(jié)合,生命周期等。
Steeltoe的組件升級(jí)到2.1后,不少示例代碼已經(jīng)使用HttpClientFactory了。當(dāng)然這是個(gè)題外話。
這里主要講的是與Polly的結(jié)合,來(lái)完成簡(jiǎn)單的熔斷降級(jí)。在這之前,還是先看看關(guān)于HttpClientFactory最簡(jiǎn)單的用法。
HttpClientFactory的簡(jiǎn)單使用
用個(gè)簡(jiǎn)單的控制臺(tái)程序來(lái)演示
這里就只是獲取一下狀態(tài)碼,沒有獲取實(shí)際的內(nèi)容。
static async Task<string> BasicUsage()
{
var serviceCollection = new ServiceCollection();
serviceCollection.AddHttpClient();
var services = serviceCollection.BuildServiceProvider();
var clientFactory = services.GetService<IHttpClientFactory>();
var client = clientFactory.CreateClient();
var request = new HttpRequestMessage(HttpMethod.Get, "https://www.github.com");
var response = await client.SendAsync(request).ConfigureAwait(false);
return response.StatusCode.ToString();
}
其實(shí)主要的操作就是AddHttpClient,然后通過HttpClientFactory創(chuàng)建一個(gè)HttpClient對(duì)象,有了HttpClient對(duì)象,下面的操作應(yīng)該就不用多說了。
然后在Main方法調(diào)用
Console.WriteLine($"BasicUsage, StatusCode = {BasicUsage().GetAwaiter().GetResult()}");
用法感覺并沒有太多的差別。下面來(lái)看看與Polly的結(jié)合。
HttpClientFactory和Polly的結(jié)合
Polly的wiki頁(yè)面已經(jīng)有了這兩者結(jié)合使用的文檔了。
https://github.com/App-vNext/Polly/wiki/Polly-and-HttpClientFactory
其實(shí)現(xiàn)在對(duì)于我們來(lái)說,要想對(duì)http請(qǐng)求使用Polly的一些特性已經(jīng)非常的簡(jiǎn)單了。
我們?cè)谑褂玫臅r(shí)候要添加Microsoft.Extensions.Http.Polly的Nuget包。
先來(lái)看看使用Polly的三種擴(kuò)展方法
| 擴(kuò)展方法 | 說明 |
|---|---|
| AddTransientHttpErrorPolicy | 主要是處理Http請(qǐng)求的錯(cuò)誤,如HTTP 5XX 的狀態(tài)碼,HTTP 408 的狀態(tài)碼 以及System.Net.Http.HttpRequestException異常。 |
| AddPolicyHandler | 自定義,和傳統(tǒng)定義Polly的方式保持一致 |
| AddPolicyHandlerFromRegistry | 從Policy集合(也是自定義的)里面選擇自己想要的。 |
后面的操作,是用的AddPolicyHandler。
由于我們要實(shí)現(xiàn)熔斷降級(jí),所以,我們必不可少的要用到CircuitBreakerPolicy和FallbackPolicy,同時(shí)為了方便演示,再加個(gè)TimeoutPolicy。
由于涉及到多個(gè)Policy,所以我們必須要確定他們的執(zhí)行順序!
Polly的wiki頁(yè)面有個(gè)示例,還配了一幅很詳細(xì)的時(shí)序圖。
一句話來(lái)說就是最先起作用的,還是最后添加的那個(gè)。
下面就新建一個(gè)API項(xiàng)目,用來(lái)演示一下。
修改ConfigureServices方法,具體如下
public void ConfigureServices(IServiceCollection services)
{
var fallbackResponse = new HttpResponseMessage();
fallbackResponse.Content = new StringContent("fallback");
fallbackResponse.StatusCode = System.Net.HttpStatusCode.TooManyRequests;
services.AddHttpClient("cb", x =>
{
x.BaseAddress = new Uri("http://localhost:8000");
x.DefaultRequestHeaders.Add("User-Agent", "HttpClientFactory-Test");
})
//fallback
.AddPolicyHandler(Policy<HttpResponseMessage>.Handle<Exception>().FallbackAsync(fallbackResponse, async b =>
{
Logger.LogWarning($"fallback here {b.Exception.Message}");
}))
//circuit breaker
.AddPolicyHandler(Policy<HttpResponseMessage>.Handle<Exception>().CircuitBreakerAsync(2, TimeSpan.FromSeconds(4), (ex, ts) =>
{
Logger.LogWarning($"break here {ts.TotalMilliseconds}");
}, () =>
{
Logger.LogWarning($"reset here ");
}))
//timeout
.AddPolicyHandler(Policy.TimeoutAsync<HttpResponseMessage>(1));
services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_1);
}
然后是在控制器去使用。
[Route("api/[controller]")]
[ApiController]
public class ValuesController : ControllerBase
{
private static int myCount = 0;
private readonly IHttpClientFactory _clientFactory;
public ValuesController(IHttpClientFactory clientFactory)
{
this._clientFactory = clientFactory;
}
// GET api/values/timeout
[HttpGet("timeout")]
public ActionResult<IEnumerable<string>> Timeout()
{
if (myCount < 3)//模擬超時(shí)
{
System.Threading.Thread.Sleep(3000);
}
myCount++;
return new string[] { "value1", "value2" };
}
// GET api/values
[HttpGet("")]
public async Task<string> GetAsync()
{
var client = _clientFactory.CreateClient("cb");
var request = new HttpRequestMessage(HttpMethod.Get, "/api/values/timeout");
var response = await client.SendAsync(request);
var content = await response.Content.ReadAsStringAsync();
return content;
}
}
效果如下

前面幾次請(qǐng)求,會(huì)因?yàn)槌瑫r(shí)或熔斷,從而我們得到的結(jié)果是fallback。
過了4秒鐘后再請(qǐng)求,由于沒有超時(shí),正常拿到了結(jié)果,所以熔斷器會(huì)被reset。
來(lái)看看日志

比較清晰的看到了所有的操作。
總結(jié)
總體來(lái)說,HttpClientFactory還是很不錯(cuò)的。尤其是它可以直接使用Polly相關(guān)的特性。
部分示例代碼: HttpClientFactoryDemo
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
- asp.net core為IHttpClientFactory添加動(dòng)態(tài)命名配置
- 如何在ASP.NET Core中使用HttpClientFactory
- 詳解如何在ASP.NET Core中使用IHttpClientFactory
- .net Core 使用IHttpClientFactory請(qǐng)求實(shí)現(xiàn)
- .Net Core下HTTP請(qǐng)求IHttpClientFactory示例詳解
- DotNetCore深入了解之HttpClientFactory類詳解
- .NET Core 2.1中HttpClientFactory的最佳實(shí)踐記錄
- .NET Core中的HttpClientFactory類用法詳解
相關(guān)文章
asp.net下數(shù)據(jù)庫(kù)操作優(yōu)化一例
數(shù)據(jù)庫(kù)升級(jí),需要對(duì)幾個(gè)表進(jìn)行一些數(shù)據(jù)轉(zhuǎn)換,具體是這樣:針對(duì)每一個(gè) Item,從 orders 表里查出 Shop_Id,并把此 Id 賦值給 items 和 skus 中的 Shop_Id。2010-11-11
ASP.NET Core文件壓縮常見使用誤區(qū)(最佳實(shí)踐)
本文給大家分享ASP.NET Core文件壓縮常見的三種誤區(qū),就每種誤區(qū)給大家講解的非常詳細(xì),是項(xiàng)目實(shí)踐的最佳紀(jì)錄,對(duì)ASP.NET Core文件壓縮相關(guān)知識(shí)感興趣的朋友一起看看吧2021-05-05
.net中自定義錯(cuò)誤頁(yè)面的實(shí)現(xiàn)升級(jí)篇
這篇文章主要給大家介紹了關(guān)于.net中自定義錯(cuò)誤頁(yè)面實(shí)現(xiàn)的相關(guān)資料,這篇文章是之前的升級(jí)篇,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-06-06
jQuery Data Linking 對(duì)象與對(duì)象之間屬性的關(guān)聯(lián)
ASP.NET團(tuán)隊(duì)最近還向jQuery社區(qū)提交了被稱為data linking的技術(shù),Data Linking可以幫助你實(shí)現(xiàn)對(duì)象與對(duì)象之間屬性的關(guān)聯(lián)——當(dāng)其中一方發(fā)生改變時(shí)另一方也隨之改變。2010-12-12
.net中實(shí)現(xiàn)listBox左右移動(dòng)
這里給大家推薦的是一段網(wǎng)友分享的,使用.net實(shí)現(xiàn)listBox左右移動(dòng)的代碼,簡(jiǎn)單實(shí)用,這里記錄下來(lái),有需要的小伙伴參考下吧。2015-03-03
javascript實(shí)現(xiàn)listbox左右移動(dòng)實(shí)現(xiàn)代碼
javascript實(shí)現(xiàn)listbox左右移動(dòng)實(shí)現(xiàn)代碼,需要的朋友可以參考下。2010-05-05
解決DropDownList總是選中第一項(xiàng)的方法
這篇文章主要介紹了解決DropDownList總是選中第一項(xiàng)的方法,DropDownList下拉框服務(wù)器控件是最常用的控件之一,雖然使用方便,但也會(huì)出現(xiàn)奇怪的錯(cuò)誤,感興趣的小伙伴們可以參考一下2015-10-10
ASP.NET Core應(yīng)用錯(cuò)誤處理之StatusCodePagesMiddleware中間件針對(duì)響應(yīng)碼呈現(xiàn)錯(cuò)誤頁(yè)面
這篇文章主要給大家介紹了關(guān)于ASP.NET Core應(yīng)用錯(cuò)誤處理之StatusCodePagesMiddleware中間件針對(duì)響應(yīng)碼呈現(xiàn)錯(cuò)誤頁(yè)面的相關(guān)資料,需要的朋友可以參考下2019-01-01

