为HttpClient添加默认请求报头的四种解决方案

所属分类: 网络编程 / ASP.NET 阅读数: 1867
收藏 0 赞 0 分享

前言

HttpClient在Web调用中具有广泛的应用,而为它添加默认请求头是我们经常遇到的需求,本文介绍4种为HttpClient添加默认请求头的方式。下面话不多说了,来一起看看详细的介绍吧

第一种方式

直接在创建的HttpClient对象的DefaultRequestHeaders集合中添加报头。

class Program
{
  static Task Main()=> SendAsync1();

  private static async Task SendAsync1()
  {
    var httpClient = new HttpClient();
    AddDefaultHeaders(httpClient);
    await httpClient.GetStringAsync("http://localhost:5000/");
  }

  private static void AddDefaultHeaders(HttpClient httpClient)
  {
    httpClient.DefaultRequestHeaders.Add("x-www-foo", "123");
    httpClient.DefaultRequestHeaders.Add("x-www-bar", "456");
    httpClient.DefaultRequestHeaders.Add("x-www-baz", "789");
  }
}

第二种方式

对于.NET Core应用来说,我们更推荐的做法是采用依赖注入的方式,利用IHttpClientFactory来创建HttpClient对象,那么我们在进行相关服务注册的时候就可以设置默认请求报头。

class Program
{
  static Task Main()=> SendAsync2();

  private static async Task SendAsync2()
  {
    var services = new ServiceCollection();
    services.AddHttpClient("", AddDefaultHeaders);
    var httpClient = services
      .BuildServiceProvider()
      .GetRequiredService<IHttpClientFactory>()
      .CreateClient();
    await httpClient.GetStringAsync("http://localhost:5000/");
  }

  private static void AddDefaultHeaders(HttpClient httpClient)
  {
    httpClient.DefaultRequestHeaders.Add("x-www-foo", "123");
    httpClient.DefaultRequestHeaders.Add("x-www-bar", "456");
    httpClient.DefaultRequestHeaders.Add("x-www-baz", "789");
  }
}

第三种方式

由于HttpClient在发送请求的时候会利用DiagnosticSource对象发送相应的诊断事件,并且将作为请求的HttpRequestMessage对象作为请求事件内容负载。我们可以订阅该事件,在请求被发送之前将其拦截下来,并添加相应的请求头即可。

class Program
{
  static Task Main()=> SendAsync3();

  private static async Task SendAsync3()
  {
    Func<object, HttpRequestMessage> requestAccessor = null;
    DiagnosticListener.AllListeners.Subscribe(listener =>
    {
      if (listener.Name == "HttpHandlerDiagnosticListener")
      {
        listener.Subscribe(kv =>
        {
          if (kv.Key == "System.Net.Http.HttpRequestOut.Start")
          {
            requestAccessor ??= BuildRequestAccessor(kv.Value.GetType());
            var request = requestAccessor(kv.Value);
            AddDefaultHeaders(request);
          }
        });
      }
    });

    var httpClient = new HttpClient();
    await httpClient.GetStringAsync("http://localhost:5000/");
    static Func<object, HttpRequestMessage> BuildRequestAccessor(Type payloadType)
    {
      var property = payloadType.GetProperty("Request", BindingFlags.Instance | BindingFlags.Public);
      var payload = Expression.Parameter(typeof(object));
      var convertedPayload = Expression.Convert(payload, payloadType);
      var getRequest = Expression.Call(convertedPayload, property.GetMethod);
      var convert = Expression.Convert(getRequest, typeof(HttpRequestMessage));
      return Expression.Lambda<Func<object, HttpRequestMessage>>(convert, payload).Compile();
    }
  }

  private static void AddDefaultHeaders(HttpRequestMessage request)
  {
    request.Headers.Add("x-www-foo", "123");
    request.Headers.Add("x-www-bar", "456");
    request.Headers.Add("x-www-baz", "789");
  }
}

第四种方式

上面这种方式可以采用强类型编程方式,具体的代码如下。

class Program
{
  static Task Main()=> SendAsync4();

  private static async Task SendAsync4()
  {
    DiagnosticListener.AllListeners.Subscribe(listener =>
    {
      if (listener.Name == "HttpHandlerDiagnosticListener")
      {
        listener.SubscribeWithAdapter(new HttpClientListener());
      }
    });
      
    var httpClient = new HttpClient();
    await httpClient.GetStringAsync("http://localhost:5000/");
  }

  private sealed class HttpClientListener
  {
    [DiagnosticName("System.Net.Http.HttpRequestOut.Start")]
    public void OnSend(HttpRequestMessage request) => AddDefaultHeaders(request);

    //Must subscribute the System.Net.Http.HttpRequestOut event.
    [DiagnosticName("System.Net.Http.HttpRequestOut")]
    public void OnSend() { }
  }

  private static void AddDefaultHeaders(HttpRequestMessage request)
  {
    request.Headers.Add("x-www-foo", "123");
    request.Headers.Add("x-www-bar", "456");
    request.Headers.Add("x-www-baz", "789");
  }
}

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。

更多精彩内容其他人还在看

开源跨平台运行服务插件TaskCore.MainForm

这篇文章主要为大家详细介绍了开源跨平台运行服务插件TaskCore.MainForm的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

CKEditor自定义按钮插入服务端图片

这篇文章主要为大家详细介绍了CKEditor自定义按钮插入服务端图片的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Asp.net Web Api实现图片点击式图片验证码功能

现在验证码的形式越来越丰富,今天要实现的是在点击图片中的文字来进行校验的验证码。下面通过本文给大家分享Asp.net Web Api实现图片点击式图片验证码功能,需要的的朋友参考下吧
收藏 0 赞 0 分享

WPF实现ScrollViewer滚动到指定控件处

这篇文章主要为大家详细介绍了WPF实现ScrollViewer滚动到指定控件处,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

WPF实现带全选复选框的列表控件

这篇文章主要为大家详细介绍了WPF实现带全选复选框的列表控件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

Asp.net MVC 中利用jquery datatables 实现数据分页显示功能

这篇文章主要介绍了Asp.net MVC 中利用jquery datatables 实现数据分页显示功能,非常不错,具有参考借鉴价值,需要的朋友可以参考下
收藏 0 赞 0 分享

asp.net 利用NPOI导出Excel通用类的方法

本篇文章主要介绍了asp.net 利用NPOI导出Excel通用类的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

VS2015自带LocalDB数据库用法详解

这篇文章主要为大家详细介绍了VS2015自带LocalDB数据库的用法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

SignalR Self Host+MVC等多端消息推送服务(一)

这篇文章主要为大家详细介绍了SignalR Self Host+MVC等多端消息推送服务,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享

SignalR Self Host+MVC等多端消息推送服务(二)

这篇文章主要为大家详细介绍了SignalR Self Host+MVC等多端消息推送服务的第二篇,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
收藏 0 赞 0 分享
查看更多