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

所属分类: 网络编程 / ASP.NET 阅读数: 1883
收藏 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");
  }
}

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

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

.NET Core源码解析配置文件及依赖注入

这篇文章我们设计了一些复杂的概念,因为要对ASP.NET Core的启动及运行原理、配置文件的加载过程进行分析,依赖注入,控制反转等概念的讲解等
收藏 0 赞 0 分享

.NET Corek中Git的常用命令及实战演练

这篇文章将通过故事的形式从Git的历史谈起,并讲述Git的强大之处。然后通过实战演练教你如何在Github以及码云上托管我们的代码并进行代码的版本控制
收藏 0 赞 0 分享

Asp.Net Core WebAPI使用Swagger时API隐藏和分组详解

这篇文章主要给大家介绍了关于Asp.Net Core WebAPI使用Swagger时API隐藏和分组的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Asp.Net Core具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
收藏 0 赞 0 分享

如何利用FluentMigrator实现数据库迁移

这篇文章主要给大家介绍了关于如何利用FluentMigrator实现数据库迁移的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
收藏 0 赞 0 分享

ASP.NET Core利用Jaeger实现分布式追踪详解

这篇文章主要给大家介绍了关于ASP.NET Core利用Jaeger实现分布式追踪的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用ASP.NET Core具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
收藏 0 赞 0 分享

浅谈从ASP.NET Core2.2到3.0你可能会遇到这些问题

这篇文章主要介绍了ASP.NET Core2.2到3.0可能会遇到的问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

详解.net core webapi 前后端开发分离后的配置和部署

这篇文章主要介绍了.net core webapi 前后端开发分离后的配置和部署,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

详解ASP.Net Core 中如何借助CSRedis实现一个安全高效的分布式锁

这篇文章主要介绍了ASP.Net Core 中如何借助CSRedis实现一个安全高效的分布式锁,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
收藏 0 赞 0 分享

.net 4.5部署到docker容器的完整步骤

这篇文章主要给大家介绍了关于.net 4.5部署到docker容器的完整步骤,文中通过示例代码介绍的非常详细,对大家学习或者使用.net4.5具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
收藏 0 赞 0 分享

.net core并发下线程安全问题详解

这篇文章主要给大家介绍了关于.net core并发下线程安全问题的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用.net core具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
收藏 0 赞 0 分享
查看更多