一、HttpClient用法
HttpClient 提供的方法:
GetAsync(String
)
GetAsync(URI
)
GetAsync(String
, HttpCompletionOption
)
GetAsync(String
, CancellationToken
)
GetAsync(Uri
, HttpCompletionOption
)
GetAsync(Uri
, HttpCompletionOption
, CancellationToken
)
GetAsync(Uri
, HttpCompletionOption
, CancellationToken
)
GetByteArrayAsync(String
)
GetByteArrayAsync(Uri
)
GetHashCode
GetStreamAsync(String
)
GetStreamAsync(Uri
)
GetStreamAsync(String
)
GetStringAsync(Uri
)
using(var httpClient
= new HttpClient())
{
}
以上用法是不推荐的,HttpClient 这个对象有点特殊,虽然继承了 IDisposable 接口,但它是可以被共享的(或者说可以被复用),且线程安全。从项目经验来看,推荐在整个应用的生命周期内复用 HttpClient 实例,而不是每次RPC请求的时候就实例化一个,在高并发的情况下,会造成Socket资源的耗尽。
1.1:基本的使用
public class Program
{
private static readonly HttpClient _httpClient
= new HttpClient();
static void Main(string[] args
)
{
HttpAsync();
Console
.WriteLine("Hello World!");
Console
.Read();
}
public static async void HttpAsync()
{
for (int i
= 0; i
< 10; i
++)
{
var result
= await _httpClient
.GetAsync("http://www.baidu.com");
if (result
.IsSuccessStatusCode
)
{
Console
.WriteLine($
"响应状态码: {(int)response.StatusCode} {response.ReasonPhrase}");
var responseBody
= await response
.Content
.ReadAsStringAsync();
Console
.WriteLine($
"响应内容:{responseBody}");
}
}
}
}
1.2:自定义SendAsync和响应头的使用
public static async Task HttpAsync()
{
try
{
using var client
= new HttpClient(new SampleMessageHandler("error"));
var response
= await client
.GetAsync(Url
);
response
.EnsureSuccessStatusCode();
ShowHeaders("响应头:", response
.Headers
);
Console
.WriteLine($
"响应状态码: {(int)response.StatusCode} {response.ReasonPhrase}");
var responseBody
= await response
.Content
.ReadAsStringAsync();
Console
.WriteLine($
"响应内容:{responseBody}");
}
catch (Exception ex
)
{
Console
.WriteLine($
"{ex.Message}");
}
}
public static void ShowHeaders(string title
, HttpHeaders headers
)
{
Console
.WriteLine(title
);
foreach (var header
in headers
)
{
var value = string.Join(" ", header
.Value
);
Console
.WriteLine($
"Header: {header.Key} Value: {value}");
}
Console
.WriteLine();
}
public class SampleMessageHandler : HttpClientHandler
{
private readonly string _displayMessage
;
public SampleMessageHandler(string message
)
{
_displayMessage
= message
;
}
protected override Task
<HttpResponseMessage
> SendAsync(HttpRequestMessage request
, CancellationToken cancellationToken
)
{
Console
.WriteLine($
"来自SampleMessageHandler的消息: {_displayMessage}");
if (_displayMessage
!= "error") return base.SendAsync(request
, cancellationToken
);
var response
= new HttpResponseMessage(HttpStatusCode
.BadRequest
);
return Task
.FromResult(response
);
}
}
1.3:HttpRequestMessage实现请求
public static async Task HttpAsync()
{
using var client
= new HttpClient();
var request
= new HttpRequestMessage(HttpMethod
.Get
, Url
);
var response
= await client
.SendAsync(request
);
if (response
.IsSuccessStatusCode
)
{
Console
.WriteLine($
"响应状态码: {(int)response.StatusCode} {response.ReasonPhrase}");
var responseBody
= await response
.Content
.ReadAsStringAsync();
Console
.WriteLine($
"响应内容 {responseBody}");
}
}
1.4:TCP实现SendAsync数据发送
public static async Task
<string> HttpAsync()
{
const int readBufferSize
= 1024;
const string hostname
= "127.0.0.1";
try
{
using var client
= new TcpClient();
await client
.ConnectAsync(hostname
, 80);
var stream
= client
.GetStream();
var header
= "GET / HTTP/1.1\r\n" +
$
"Host: {hostname}:80\r\n" +
"Connection: close\r\n" +
"\r\n";
var buffer
= Encoding
.UTF8
.GetBytes(header
);
await stream
.WriteAsync(buffer
, 0, buffer
.Length
);
await stream
.FlushAsync();
var ms
= new MemoryStream();
buffer
= new byte[readBufferSize
];
var read
= 0;
do
{
read
= await stream
.ReadAsync(buffer
, 0, readBufferSize
);
ms
.Write(buffer
, 0, read
);
Array
.Clear(buffer
, 0, buffer
.Length
);
} while (read
> 0);
ms
.Seek(0, SeekOrigin
.Begin
);
using var reader
= new StreamReader(ms
);
return reader
.ReadToEnd();
}
catch (SocketException ex
)
{
Console
.WriteLine(ex
.Message
);
return null;
}
}
HttpClient到底层的执行过程图
二、HttpClient高级用法
public async Task
<string> GetAccessTokenAsync()
{
string uri
= "你的URL";
HttpClientHandler handler
= new HttpClientHandler
{
UseDefaultCredentials
= false
};
HttpClient httpClient
= new HttpClient(handler
);
HttpResponseMessage response
= await httpClient
.GetAsync(uri
);
response
.EnsureSuccessStatusCode();
string resp
= await response
.Content
.ReadAsStringAsync();
JObject json
= (JObject
)JsonConvert
.DeserializeObject(resp
);
string accessToken
= json
["access_token"].ToString();
PostMailAsync(accessToken
);
return "success";
}
优化:帮HttpClient预热 我们采用一种预热方式,在正式发post请求之前,先发一个head请求:
_httpClient
.SendAsync(new HttpRequestMessage {
Method
= new HttpMethod("HEAD"),
RequestUri
= new Uri(BASE_ADDRESS
+ "/")
})
.Result
.EnsureSuccessStatusCode();
经测试,通过这种热身方法,可以将第一次请求的耗时由2s左右降到1s以内(测试结果是700多ms)。
存在问题 复用 HttpClient 后,依然存在一些问题:
因为是复用的 HttpClient ,那么一些公共的设置就没办法灵活的调整了,如请求头的自定义。因为 HttpClient 请求每个 url 时,会缓存该 url 对应的主机 ip ,从而会导致 DNS 更新失效( TTL 失效了) 那么有没有办法解决HttpClient的这些个问题?直到 HttpClientFactory 的出现,这些坑 “完美” 规避掉了。
三、HttpClientFactory
HttpClientFacotry 很高效,可以最大程度上节省系统 socket 。(“JUST USE IT AND FXXK SHUT UP”😛)Factory,顾名思义 HttpClientFactory 就是 HttpClient 的工厂,内部已经帮我们处理好了对 HttpClient 的管理,不需要我们人工进行对象释放,同时,支持自定义请求头,支持DNS更新等等等。从微软源码分析,HttpClient 继承自 HttpMessageInvoker ,而 HttpMessageInvoker 实质就是 HttpClientHandler 。HttpClientFactory 创建的 HttpClient ,也即是 HttpClientHandler ,这些个 HttpClient 被放到了“池子”中,工厂每次在 create 的时候会自动判断是新建还是复用。(默认生命周期为 2 min)
3.1.1在 Startup.cs 中进行注册
public class Startup
{
public Startup(IConfiguration configuration
)
{
Configuration
= configuration
;
}
public IConfiguration Configuration
{ get; }
public void ConfigureServices(IServiceCollection services
)
{
services
.AddHttpClient("client_1", config
=>
{
config
.BaseAddress
= new Uri("http://client_1.com");
config
.DefaultRequestHeaders
.Add("header_1", "header_1");
});
services
.AddHttpClient("client_2", config
=>
{
config
.BaseAddress
= new Uri("http://client_2.com");
config
.DefaultRequestHeaders
.Add("header_2", "header_2");
});
services
.AddHttpClient();
services
.AddMvc().AddFluentValidation();
}
}
3.1.2使用,这里直接以 controller 为例,其他地方自行 DI
public class TestController : ControllerBase
{
private readonly IHttpClientFactory _httpClient
;
public TestController(IHttpClientFactory httpClient
)
{
_httpClient
= httpClient
;
}
public async Task
<ActionResult
> Test()
{
var client
= _httpClient
.CreateClient("client_1");
var result
= await client
.GetStringAsync("/page1.html");
var client2
= _httpClient
.CreateClient();
var result2
= await client
.GetStringAsync("http://www.site.com/XXX.html");
return null;
}
}
3.2.1:使用自定义类执行 HttpClientFactory 请求 自定义 HttpClientFactory 请求类
public class SampleClient
{
public HttpClient Client
{ get; private set; }
public SampleClient(HttpClient httpClient
)
{
httpClient
.BaseAddress
= new Uri("https://api.SampleClient.com/");
httpClient
.DefaultRequestHeaders
.Add("Accept", "application/json");
httpClient
.DefaultRequestHeaders
.Add("User-Agent", "HttpClientFactory-Sample");
Client
= httpClient
;
}
}
3.2.2在 Startup.cs 中 ConfigureService 方法中注册 SampleClient
services
.AddHttpClient<ISampleClient, SampleClient>();
3.3.3调用:
public class ValuesController : Controller
{
private readonly ISampleClient _sampleClient
;
public ValuesController(ISampleClient sampleClient
)
{
_sampleClient
= sampleClient
;
}
[HttpGet]
public async Task
<ActionResult
> Get()
{
string result
= await _sampleClient
.GetData();
return Ok(result
);
}
}