为了账号安全,请及时绑定邮箱和手机立即绑定

用HttpClientFactory来实现简单的熔断降级

标签:
C#

前言

在2.1之后,有不少新东西,其中HttpClientFactory算是一个。HttpClientFactory涉及的东西也不算少,三四种clients , 请求中间件,与Polly的结合,生命周期等。

Steeltoe的组件升级到2.1后,不少示例代码已经使用HttpClientFactory了。当然这是个题外话。

这里主要讲的是与Polly的结合,来完成简单的熔断降级。在这之前,还是先看看关于HttpClientFactory最简单的用法。

HttpClientFactory的简单使用

用个简单的控制台程序来演示

这里就只是获取一下状态码,没有获取实际的内容。

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();
}

其实主要的操作就是AddHttpClient,然后通过HttpClientFactory创建一个HttpClient对象,有了HttpClient对象,下面的操作应该就不用多说了。

然后在Main方法调用

Console.WriteLine($"BasicUsage, StatusCode = {BasicUsage().GetAwaiter().GetResult()}");

用法感觉并没有太多的差别。下面来看看与Polly的结合。

HttpClientFactory和Polly的结合

Polly的wiki页面已经有了这两者结合使用的文档了。

https://github.com/App-vNext/Polly/wiki/Polly-and-HttpClientFactory

其实现在对于我们来说,要想对http请求使用Polly的一些特性已经非常的简单了。

我们在使用的时候要添加Microsoft.Extensions.Http.Polly的Nuget包。

先来看看使用Polly的三种扩展方法

扩展方法说明
AddTransientHttpErrorPolicy主要是处理Http请求的错误,如HTTP 5XX 的状态码,HTTP 408 的状态码 以及System.Net.Http.HttpRequestException异常。
AddPolicyHandler自定义,和传统定义Polly的方式保持一致
AddPolicyHandlerFromRegistry从Policy集合(也是自定义的)里面选择自己想要的。

后面的操作,是用的AddPolicyHandler

由于我们要实现熔断降级,所以,我们必不可少的要用到CircuitBreakerPolicy和FallbackPolicy,同时为了方便演示,再加个TimeoutPolicy。

由于涉及到多个Policy,所以我们必须要确定他们的执行顺序!

Polly的wiki页面有个示例,还配了一幅很详细的时序图。

一句话来说就是最先起作用的,还是最后添加的那个。

下面就新建一个API项目,用来演示一下。

修改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)//模拟超时
        {
            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;
    }
}

效果如下

https://img1.sycdn.imooc.com//5b5491c80001a43906870065.jpg

前面几次请求,会因为超时或熔断,从而我们得到的结果是fallback。

过了4秒钟后再请求,由于没有超时,正常拿到了结果,所以熔断器会被reset。

来看看日志

https://img1.sycdn.imooc.com//5b5491dc0001b3bc11401462.jpg

比较清晰的看到了所有的操作。

总结

总体来说,HttpClientFactory还是很不错的。尤其是它可以直接使用Polly相关的特性。

部分示例代码: HttpClientFactoryDemo

如果您认为这篇文章还不错或者有所收获,可以点击右下角的【推荐】按钮,因为你的支持是我继续写作,分享的最大动力!

作者:Catcher ( 黄文清 )

来源:http://catcher1994.cnblogs.com/

声明: 本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。如果您发现博客中出现了错误,或者有更好的建议、想法,请及时与我联系!!如果想找我私下交流,可以私信或者加我QQ。

原文出处


点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消