• NET 5 使用HttpClient和HttpWebRequest


    HttpWebRequest

    这是.NET创建者最初开发用于使用HTTP请求的标准类。HttpWebRequest是老版本.net下常用的,较为底层且复杂,访问速度及并发也不甚理想,但是使用HttpWebRequest可以让开发者控制请求/响应流程的各个方面,如 timeouts, cookies, headers, protocols。另一个好处是HttpWebRequest类不会阻塞UI线程。例如,当您从响应很慢的API服务器下载大文件时,您的应用程序的UI不会停止响应。通常和WebResponse一起使用,一个发送请求,一个获取数据。另外HttpWebRequest库已经过时,不适合业务中直接使用,他更适用于框架内部操作。

    /// <summary>
            /// HttpWebRequest请求网页示例
            /// </summary>
            /// <param name="args"></param>
            static void Main(string[] args)
            {
                HttpWebRequest httpWebRequest = null;
                HttpWebResponse httpWebResponse = null;
                Stream responseStream = null;
                string url = "https://www.cnblogs.com/";
                try
                {
                    httpWebRequest = (HttpWebRequest)HttpWebRequest.Create(url);
    
                    //cookie,cookie一般用来验证登录或是跟踪使用
                    httpWebRequest.CookieContainer = new CookieContainer();
                    httpWebRequest.CookieContainer.Add(new Cookie() { Name = "test", Value = "test1",Domain="www.cnblogs.com" });
    
                    //来源页面
                    httpWebRequest.Referer = url;
    
                    //比较重要的UserAgent
                    httpWebRequest.UserAgent = "Mozilla/5.0 (Windows NT 10.0; WOW64; rv:52.0 Gecko/20100101 Firefox/52.0";
    
                    //请求方法,有GET,POPST,PUT等
                    httpWebRequest.Method = "GET";
    
                    //如果上传文件,是要设置 GetRequestStream
                    //httpWebRequest.GetRequestStream
    
                    try
                    {
                        httpWebResponse = (HttpWebResponse)httpWebRequest.GetResponse();
                    }
                    catch (System.Net.WebException we)
                    {
                        ///这个说明服务器返回了信息了,不过是非200,301,302这样正常的状态码
                        if (we.Response != null)
                        {
                            httpWebResponse = (HttpWebResponse)we.Response;
                        }
                    }
    
                    ///得到返回的stream,如果请求的是一个文件或图片,可以直接使用或保存
                    responseStream = httpWebResponse.GetResponseStream();
    
                    ///使用utf8方式读取数据流
                    StreamReader streamReader = new StreamReader(responseStream, Encoding.UTF8);
    
                    ///这里是一次性读取,对于超大的stream,要不断读取并保存
                    string html = streamReader.ReadToEnd();
                    streamReader.Close();
                    responseStream.Close();
                    Console.WriteLine(html.Length);
                }
                catch (Exception ex)
                {
                    Console.WriteLine(ex.Message);
                }
                finally
                {
                    if (httpWebRequest != null) httpWebRequest.Abort();
                    if (httpWebResponse != null) httpWebResponse.Close();
                    if (responseStream != null) responseStream.Close();
                }
            }
    View Code

    HttpClient

    HttpClient提供强大的功能,提供了异步支持,可以轻松配合async await 实现异步请求,使用HttpClient,在并发量不大的情况,一般没有任何问题;但是在并发量一上去,如果使用不当,会造成很严重的堵塞的情况。

    平时我们在使用HttpClient的时候,会将HttpClient包裹在using内部进行声明和初始化,

    using(var httpClient = new HttpClient())
    {
        //other codes
    }

    在高并发的情况下,连接来不及释放,socket被耗尽,耗尽之后就会出现喜闻乐见的一个错误:

    Unable to connect to the remote serverSystem.Net.Sockets.SocketException: Only one usage of each socket address (protocol/network address/port) is normally permitted.

    那么如何处理这个问题?“复用HttpClient”即可

    • HttpClientFacotry很高效,可以最大程度上节省系统socket。(“JUST USE IT AND FXXK SHUT UP”:P)
    • Factory,顾名思义HttpClientFactory就是HttpClient的工厂,内部已经帮我们处理好了对HttpClient的管理,不需要我们人工进行对象释放,同时,支持自定义请求头,支持DNS更新等等等

    从微软源码分析,HttpClient继承自HttpMessageInvoker,而HttpMessageInvoker实质就是HttpClientHandler。

    HttpClientFactory 创建的HttpClient,也即是HttpClientHandler,只是这些个HttpClient被放到了“池子”中,工厂每次在create的时候会自动判断是新建还是复用。(默认生命周期为2min)

    还理解不了的话,可以参考Task和Thread的关系

    解决方案如下:

    IHttpClientFactory

    一、可以参考微软官方提供的方法:https://docs.microsoft.com/zh-cn/aspnet/core/fundamentals/http-requests?view=aspnetcore-3.1

    https://docs.microsoft.com/zh-cn/aspnet/core/fundamentals/http-requests?view=aspnetcore-5.0

    二、我的解决方案是根据官方提供的方法,选择一种最适合项目的写法进行改造。

    1、nuget添加包Microsoft.AspNetCore.Http;

    2、startup里ConfigureServices方法添加代码:

    services.AddHttpClient();

    or

    public void ConfigureServices(IServiceCollection services)
            {
                //other codes
                
                services.AddHttpClient("client_1",config=>  //这里指定的name=client_1,可以方便我们后期服用该实例 比如已经填写url和header
                {
                    config.BaseAddress= new Uri("http://client_1.com");
                    config.DefaultRequestHeaders.Add("header_1","header_1");            });
    
                services.AddHttpClient();
    
                //other codes
                services.AddMvc().AddFluentValidation();
            }

    3、可以使用依赖项注入 (DI) 来请求 IHttpClientFactory。 以下代码使用 IHttpClientFactory 来创建 HttpClient 实例:(官方demo)

    public class BasicUsageModel : PageModel
    {
        private readonly IHttpClientFactory _clientFactory;
    
        public IEnumerable<GitHubBranch> Branches { get; private set; }
    
        public bool GetBranchesError { get; private set; }
    
        public BasicUsageModel(IHttpClientFactory clientFactory)
        {
            _clientFactory = clientFactory;
        }
    
        public async Task OnGet()
        {
            var request = new HttpRequestMessage(HttpMethod.Get,
                "https://api.github.com/repos/aspnet/AspNetCore.Docs/branches");
            request.Headers.Add("Accept", "application/vnd.github.v3+json");
            request.Headers.Add("User-Agent", "HttpClientFactory-Sample");
    
            var client = _clientFactory.CreateClient();
    
            var response = await client.SendAsync(request);
    
            if (response.IsSuccessStatusCode)
            {
                using var responseStream = await response.Content.ReadAsStreamAsync();
                Branches = await JsonSerializer.DeserializeAsync
                    <IEnumerable<GitHubBranch>>(responseStream);
            }
            else
            {
                GetBranchesError = true;
                Branches = Array.Empty<GitHubBranch>();
            }
        }
    }

    在实际使用中,我们经常会用NewtonJson序列化,给一个简单的Demo:

    string api_domain = _config.GetSection("OuterApi:open-api").Value;
                    string api_url = $"{api_domain}/common-service/api/basic?code={code}";
                    var request = new HttpRequestMessage(HttpMethod.Get, api_url);
                    request.Headers.Add("Accept", "application/vnd.github.v3+json");
    
    
                    var client = _clientFactory.CreateClient();
    
                    var response = await client.SendAsync(request);
    
                    Result<List<OpenApiDictModel>> apiRet = new Result<List<OpenApiDictModel>>();
                    if (response.IsSuccessStatusCode)
                    {
                        string responseStr = await response.Content.ReadAsStringAsync();
                        apiRet = JsonConvert.DeserializeObject<Result<List<OpenApiDictModel>>>(responseStr);
                    }
  • 相关阅读:
    BAT 批处理 for循环 迟环境变量 [MD]
    adb 环境配置 常用命令 [MD]
    XML 解析 DOM SAX PULL 序列化 总结 [MD]
    instanceof 和 isInstance 强转 类型 class [MD]
    Charles 简介 总结 HTTP 抓包 代理 [MD]
    Permission 运行时权限 总结 翻译 [MD]
    反编译 AndroidKiller 逆向 字节码 实践案例 [MD]
    Gradle 翻译 Analyzer APK文件分析 [MD]
    Java 中 boolean 类型占用多少个字节 [MD]
    Shell 命令行工具 Cmder Babun Zsh [MD]
  • 原文地址:https://www.cnblogs.com/netlock/p/14101161.html
Copyright © 2020-2023  润新知