為HttpClient添加默認(rèn)請(qǐng)求報(bào)頭的四種解決方案
前言
HttpClient在Web調(diào)用中具有廣泛的應(yīng)用,而為它添加默認(rèn)請(qǐng)求頭是我們經(jīng)常遇到的需求,本文介紹4種為HttpClient添加默認(rèn)請(qǐng)求頭的方式。下面話不多說了,來一起看看詳細(xì)的介紹吧
第一種方式
直接在創(chuàng)建的HttpClient對(duì)象的DefaultRequestHeaders集合中添加報(bào)頭。
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");
}
}
第二種方式
對(duì)于.NET Core應(yīng)用來說,我們更推薦的做法是采用依賴注入的方式,利用IHttpClientFactory來創(chuàng)建HttpClient對(duì)象,那么我們?cè)谶M(jìn)行相關(guān)服務(wù)注冊(cè)的時(shí)候就可以設(shè)置默認(rèn)請(qǐng)求報(bào)頭。
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在發(fā)送請(qǐng)求的時(shí)候會(huì)利用DiagnosticSource對(duì)象發(fā)送相應(yīng)的診斷事件,并且將作為請(qǐng)求的HttpRequestMessage對(duì)象作為請(qǐng)求事件內(nèi)容負(fù)載。我們可以訂閱該事件,在請(qǐng)求被發(fā)送之前將其攔截下來,并添加相應(yīng)的請(qǐng)求頭即可。
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");
}
}
第四種方式
上面這種方式可以采用強(qiáng)類型編程方式,具體的代碼如下。
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");
}
}
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Entity Framework使用Code First模式管理視圖
本文詳細(xì)講解了Entity Framework使用Code First模式管理視圖的方法,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-03-03
WPF制作一個(gè)簡(jiǎn)單的倒計(jì)時(shí)器實(shí)例附源碼
既然早上沒事干,于是想到前些日子學(xué)院的某個(gè)老師讓大家給他找個(gè)什么倒計(jì)時(shí)的小軟件;何不寫個(gè)玩玩~既然要寫,就用以前沒怎么搗鼓過的WPF寫一個(gè)倒計(jì)時(shí)器,需要了解的朋友可以參考下2012-12-12
asp.net基于HashTable實(shí)現(xiàn)購(gòu)物車的方法
這篇文章主要介紹了asp.net基于HashTable實(shí)現(xiàn)購(gòu)物車的方法,涉及asp.net中HashTable結(jié)合session實(shí)現(xiàn)購(gòu)物車功能的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-12-12
C#中efcore-ShardingCore呈現(xiàn)“完美”分表
本文簡(jiǎn)單的介紹了efcore的分表,著重介紹了efcore下最完美的分表組件ShardingCore,可以幫助大家更好的學(xué)習(xí),感興趣的小伙伴可以參考一下2021-08-08
asp.net操作javascript:confirm返回值的兩種方式
asp.net操作javascript:confirm返回值分為兩種,不使用ajax、使用了ajax,不使用ajax,可以用StringBuilder來完成2014-09-09
asp.NET中實(shí)現(xiàn)文件的壓縮和解壓(3種方式)
本篇文章主要介紹了asp.NET中實(shí)現(xiàn)文件的壓縮和解壓,這里整理了詳細(xì)的代碼,有需要的小伙伴可以參考下。2016-11-11

