Java9之HttpClientAPI实战详解

news2024/9/22 3:31:25

Java9 之 HttpClientAPI 实战详解

前言

相信关注 java9 的小伙伴们都知道 java9 版本内置模块提供了 Http 功能,当然并不是说之前 jdk 之前并不支持,那么这次更新又多了什么呢?或者是解决了什么问题?

说明

自 JDK 1.0 以来,Java 已经支持 HTTP/1.1。 HTTP API 由 java.net 包中的几种类型组成。 现有的 API 有以下问题:

  • 它被设计为支持多个协议,如 http,ftp,gopher 等,其中许多协议不再被使用。
  • 太抽象了,很难使用。
  • 它包含许多未公开的行为。
  • 它只支持一种模式,阻塞模式,这要求每个请求 / 响应有一个单独的线程。

2015 年 5 月,IETF(Internet Engineering Task Force)发布了 HTTP/2 规范。 有关 HTTP/2 规范的完整文本,请访问 https://tools.ietf.org/html/rfc7540。 HTTP/2 不会修改应用程序级语义。 也就是说,对应用程序中的 HTTP 协议的了解和使用情况并没有改变。 它具有更有效的方式准备数据包,然后发送到客户端和服务器之间的电线。 所有之前知道的 HTTP,如 HTTP 头,方法,状态码,URL 等都保持不变。 HTTP/2 尝试解决与 HTTP/1 连接所面临的许多性能相关的问题:

  • HTTP/2 支持二进制数据交换,来代替 HTTP/1.1 支持的文本数据。
  • HTTP/2 支持多路复用和并发,这意味着多个数据交换可以同时发生在 TCP 连接的两个方向上,而对请求的响应可以按顺序接收。 这消除了在对等体之间具有多个连接的开销,这在使用 HTTP/1.1 时通常是这种情况。 在 HTTP/1.1 中,必须按照发送请求的顺序接收响应,这称为 head-of-line 阻塞。 HTTP/2 通过在同一 TCP 连接上进行复用来解决线路阻塞问题。
  • 客户端可以建议请求的优先级,服务器可以在对响应进行优先级排序时予以遵守。
  • HTTP 首部(header)被压缩,这大大降低了首部大小,从而降低了延迟。
  • 它允许从服务器到客户端的资源推送。

JDK 9 不是更新现有的 HTTP/1.1 API,而是提供了一个支持 HTTP/1.1 和 HTTP/2 的 HTTP/2 Client API。 该 API 旨在最终取代旧的 API。 新 API 还包含使用 WebSocket 协议开发客户端应用程序的类和接口。 有关完整的 WebSocket 协议规范,请访问 https://tools.ietf.org/html/rfc6455。新的 HTTP/2 客户端 API 与现有的 API 相比有以下几个好处:

  • 在大多数常见情况下,学习和使用简单易用。
  • 它提供基于事件的通知。 例如,当收到首部信息,收到正文并发生错误时,会生成通知。
  • 它支持服务器推送,这允许服务器将资源推送到客户端,而客户端不需要明确的请求。 它使得与服务器的 WebSocket 通信设置变得简单。
  • 它支持 HTTP/2 和 HTTPS/TLS 协议。
  • 它同时工作在同步(阻塞模式)和异步(非阻塞模式)模式。

如果想使用 Java9 的 HttpClient 服务,那么你必须熟悉(jdk.incubator.http)包中的以下三个类:

HttpClient http 客户端

该类是 Java9 开始引入的,官方文档的翻译说明是这样的

  • HttpClient 是一个对多个请求配置了公共信息的容器。所有的请求通过一个 HttpClient 进行发送。HttpClients 是不可变的,通过 HttpClient 的 newBuilder () 创建返回。请求 Builders 被 HttpRequest#newBuilder () 来创建。
  • 接口 API

API 中 Builder 部分用来构建客户端的配置,send 相关的几个方法是进行请求发送,不同的是 Async 是异步操作。其他的基本是客户端的参数配置信息 (包括代理,线程,版本,SSL,cookie 等),同时也提供了 socket 支持。

  • 使用示例

    • 示例 1,使用默认配置

    HttpClient client = HttpClient.newHttpClient();

    • 示例 2, 自定义配置。

    try {
    Authenticator authenticator=new Authenticator() {
    };
    client= HttpClient.newBuilder()
    .authenticator(authenticator)//配置authenticator
    .sslContext(SSLContext.getDefault())//配置 sslContext
    .sslParameters(new SSLParameters())//配置 sslParameters
    .proxy(ProxySelector.getDefault())//配置 proxy
    .executor(Executors.newCachedThreadPool())//配置 executor
    .followRedirects(HttpClient.Redirect.ALWAYS)//配置 followRedirects
    .cookieManager(new CookieManager())//配置 cookieManager
    .version(HttpClient.Version.HTTP_2)//配置 version
    .build();

        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        }
    
  • 说明

    由于 HttpClient 隶属于 jdk.incubator.httpclient,所以使用的时候需要添加模块依赖方可执行。 如果你是单个 class,没有引入模块概念的话需要在 VM 参数中添加模块支持 --add-modules jdk.incubator.httpclient。如果你引入了模块的概念,需要在 你的 module.info 中添加 requires jdk.incubator.httpclient; 依赖。

HttpRequest 请求

  • API 文档说明

表示可以发送到服务器的一个 HTTP 请求。 HttpRequest 由 HttpRequest builders 构建生成。 HttpRequest 通过调用 HttpRequest.newBuilder 获得实例。 一个请求的 URI ,head 和 body 都可以设置。 请求体提供了 HttpRequest.BodyProcessor 对象的 DELETE , POST 或 PUT 方法。 GET 不用设置 body。 一旦所有必需的参数都在构建器设置, HttpRequest.Builder.build () 将返回一个 HttpRequest 实例 。 构建器也可以被多次复制和修改,以构建参数不同的多个相关请求。

  • 使用示例

    • 示例 1,GET 请求

    HttpResponse<String> response = client.send(
    HttpRequest
    .newBuilder(new URI(“http://www.foo.com/”))
    .headers(“Foo”, “foovalue”, “Bar”, “barvalue”)
    .GET()
    .build(),
    BodyHandler.asString()
    );
    int statusCode = response.statusCode();
    String body = response.body();

    • 示例 2,POST 请求。

    HttpResponse response = client.send(
    HttpRequest
    .newBuilder(new URI(“http://www.foo.com/”))
    .headers(“Foo”, “foovalue”, “Bar”, “barvalue”)
    .POST(BodyProcessor.fromString(“Hello world”))
    .build(),
    BodyHandler.asFile(Paths.get(“/path”))
    );
    int statusCode = response.statusCode();
    Path body = response.body(); // should be “/path”
    }

HttpResponse 响应

  • API 文档说明

表示 HttpRequest 的响应。 通常在响应正文,响应状态代码和 headers 被接收之后,HttpResponse 才是可用的。 这取决于发送请求时提供的响应体处理程序。 在所有情况下,在 Body 被读取之前调用 response body handler 程序。 此类中提供了访问响应头和响应主体的方法。

响应处理程序和处理器

Response bodies 有两种处理方式。 应用程序代码提供响应处理程序( HttpResponse.BodyHandler ), 一个是可以检查响应状态代码和头文件, 一个是返回一个 HttpResponse.BodyProcessor 以实际读取(或丢弃)正文并将其转换为一些有用的 Java 对象类型。 处理程序可以返回预定义的处理器类型之一或定制处理器, 如果正文被丢弃,则可以调用 BodyProcessor.discard () 并返回丢弃响应正文的处理器。 处理器和处理器的静态实现分别在 BodyHandler 和 BodyProcessor 中提供。 在所有情况下,提供的处理程序功能都是方便的实现, 它忽略了提供的状态代码和头文件,并返回相关的预定义的 BodyProcessor 。

  • 使用示例

    • 示例 1,BodyHandler

    HttpResponse resp = HttpRequest
    .create(URI.create(“http://www.foo.com”))
    .GET()
    .response(BodyHandler.asFile(Paths.get(“/tmp/f”)));
    }

    • 示例 2,BodyProcessor。

    HttpResponse resp1 = HttpRequest
    .create(URI.create(“http://www.foo.com”))
    .GET()
    .response(
    (status, headers) -> status == 200
    ? BodyProcessor.asFile(Paths.get(“/tmp/f”))
    : BodyProcessor.discard(Paths.get(“/NULL”)));
    }

实战应用

这里是一个完整的示例应用,涵盖了 HttpClient,httpRequest,HttpResponse 等的使用。

package com.javanine.http;

import jdk.incubator.http.HttpClient;
import jdk.incubator.http.HttpRequest;
import jdk.incubator.http.HttpResponse;
import jdk.incubator.http.MultiMapResult;

import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLParameters;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.net.*;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.security.NoSuchAlgorithmException;
import java.time.Duration;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Executors;

/**
 * Created by bgt on 2017/10/1.
 * Java9http示例
 * <p>
 * VM 参数中添加模块支持
 * --add-modules jdk.incubator.httpclient
 */
public class HttpDemo {
   // private static HttpClient client = HttpClient.newHttpClient();
    private static HttpClient client = null;

    static {
    //这里是httpclient的配置
      init();


    }

    public static void main(String[] args) {
       // HttpDemo.HttpGet();
        HttpDemo.HttpGet2();
    }

    public static void init(){
        try {
            Authenticator authenticator=new Authenticator() {
            };
        client= HttpClient.newBuilder()
                .authenticator(authenticator)//配置authenticator
                .sslContext(SSLContext.getDefault())//配置 sslContext
                .sslParameters(new SSLParameters())//配置 sslParameters
                .proxy(ProxySelector.getDefault())//配置 proxy
                .executor(Executors.newCachedThreadPool())//配置 executor
                .followRedirects(HttpClient.Redirect.ALWAYS)//配置 followRedirects
                .cookieManager(new CookieManager())//配置 cookieManager
                .version(HttpClient.Version.HTTP_2)//配置 version
                .build();

    } catch (NoSuchAlgorithmException e) {
        e.printStackTrace();
    }
    }
    /**
     * 普通get方式
     *
     *
     * 结果:{
     "msg" : "未查询到用户,请认真检查账户或者是否登录",
     "success" : false
     }
     响应码:200

     */
    public static void HttpGet() {

        HttpRequest request = HttpRequest
                //.newBuilder(URI.create("http://blog.csdn.net/u014042066"))
                .newBuilder(URI.create("http://www.rqbao.com/lotteryAward/gettenrecordlist"))
                .header("refer", "http://www.oschina.com")//携带的参数
                .header("userId", "d3e750db32004972b0ae58f8129f50fc")
                .timeout(Duration.ofSeconds(2))//2秒过期
                .GET()
                .build();
        getReponse(request);

    }

    public static void HttpGet2() {

        HttpRequest request = HttpRequest
                //.newBuilder(URI.create("http://blog.csdn.net/u014042066"))
                .newBuilder(URI.create("http://www.rqbao.com/lotteryAward/gettenrecordlist"))
                .header("refer", "http://www.oschina.com")//携带的参数
                .header("userId", "d3e750db32004972b0ae58f8129f50fc")
                .timeout(Duration.ofSeconds(2))//2秒过期
                .GET()
                .build();
        getAsyReponse2(request);

    }

    /**
     * 文件上传
     */
    public static void HttpPost() {
        try {
            HttpRequest request = HttpRequest
                    .newBuilder(URI.create("http://blog.csdn.net/u014042066"))
                    .header("refer", "http://www.oschina.com")
                    .POST(HttpRequest.BodyProcessor.fromFile(Paths.get("/url.txt")))
                    .build();
            getAsyReponse(request);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }

    }

    /**
     * 携带参数
     */
    public static void HttpPost2() {
        HttpRequest request = HttpRequest
                .newBuilder(URI.create("http://blog.csdn.net/u014042066"))
                .header("refer", "http://www.oschina.com")
                .POST(HttpRequest.BodyProcessor.fromString("name=ricky,pwd=123456"))
                .build();
        getReponse(request);
    }

    /**
     * 输出响应结果
     * @param request
     * @throws IOException
     * @throws InterruptedException
     */
    public static void getReponse(HttpRequest request) {
        HttpResponse<String> response = null;
        try {
            if (client==null) {
                init();
            }
            response = client.send(request, HttpResponse.BodyHandler.asString());
        } catch (IOException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        String result = response.body();
        int code = response.statusCode();
        if (code == 200) {
            System.out.println("结果:" + result);
        }
        System.out.println("响应码:" + code);
    }

    /**
     * 输出响应结果 带path形式
     * @param request
     * @throws IOException
     * @throws InterruptedException
     */
    public static void getReponsePath(HttpRequest request) {
        HttpResponse<Path> response = null;
        try {
            client.send(request, HttpResponse.BodyHandler.asFile(Paths.get("/url.text")));
            Path body = response.body();
            System.out.println(body);
        } catch (IOException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    /**
     * 输出响应结果 带path形式的异步
     * @param request
     * @throws IOException
     * @throws InterruptedException
     */
    public static void getAsyReponsePath(HttpRequest request) {
        CompletableFuture<Path> response = null;
        client
                .sendAsync(request, HttpResponse.BodyHandler.asFile(Paths.get("/url.text")))
                .thenApply((response1) -> response1.body());

        response.join();
    }

    /**
     * 异步的输出响应结果
     * @param request
     * @throws IOException
     * @throws InterruptedException
     */
    public static void getAsyReponse(HttpRequest request) {
        CompletableFuture<HttpResponse<String>> cf;
        cf = client.sendAsync(request, HttpResponse.BodyHandler.asString());
        HttpResponse<String> response = cf.join();
        System.out.println(response.body());
    }
    /**
     * 异步的输出响应结果
     * @param request
     * @throws IOException
     * @throws InterruptedException
     */
    public static void getAsyReponse2(HttpRequest request) {
        MultiMapResult<String> ress;
        ress = client.sendAsync(request, HttpResponse.MultiProcessor.asMap((req)-> Optional.of(
                HttpResponse.BodyHandler.asString()
        ))).join();
      ress.forEach((req,cf)->{
          HttpResponse<String> resp=cf.join();
          System.out.println("uri:"+resp.uri()+"---body:"+resp.body());
      });
    }
}

Java9 的 HttpClient 提供了丰富的功能,这里只是做了个简单的入门与分享,不对之处还望指正。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/377647.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

图灵完备及TypeScript图灵完备性验证

一、图灵完备 1.图灵完备的概念 首先从定义出发&#xff0c;什么是图灵完备&#xff1a;图灵完备指一系列操作数据的规则能够模拟任何图灵机。 WikiPedia-图灵完备介绍&#xff0c;在可计算性理论&#xff0c;如果一系列操作数据的规则&#xff08;如指令集、编程语言、细胞自…

centos安装Anaconda3

目录一、参考二、Anaconda简介1、用途2、关于anaconda三、下载安装1、下载2、安装anaconda3、配置环境遍历4、测试配置结果5、设置显示前缀一、参考 在centos上安装Anaconda 最新Anaconda3的安装配置及使用教程&#xff08;附图文&#xff09; 二、Anaconda简介 一句话&…

系统升级丨分享返佣,助力商企实现低成本高转化营销

秉承助力传统经济数字化转型的长远理念 酷雷曼VR再次在VR全景营销中发力 创新研发“分享返佣”功能 进一步拓宽商企VR全景营销渠道 助力商企搭建低成本、高传播、高转化 的VR营销体系 01、什么是“分享返佣”&#xff1f; ●“分享返佣”即“推广”返佣&#xff0c;是酷…

干货满满!MES的简介和运用

导读 谈及MES必须先谈生产&#xff0c;生产体系模型如图所示&#xff0c;涉及人、财、物、信息等资源&#xff0c;产、供、销等环节&#xff0c;以及供应商、客户、合作伙伴等。 其中&#xff0c;生产管理是通过对生产系统的战略计划、组织、指挥、实施、协调、控制等活动&…

【经验总结】10年的嵌入式开发老手,到底是如何快速学习和使用RT-Thread的?(文末赠书5本)

【经验总结】一位近10年的嵌入式开发老手&#xff0c;到底是如何快速学习和使用RT-Thread的&#xff1f; RT-Thread绝对可以称得上国内优秀且排名靠前的操作系统&#xff0c;在嵌入式IoT领域一直享有盛名。近些年&#xff0c;物联网产业的大热&#xff0c;更是直接将RT-Thread这…

信贷系统学习总结(5)—— 简单的风控示例(含代码)

一、背景1.为什么要做风控?目前我们业务有使用到非常多的AI能力,如ocr识别、语音测评等,这些能力往往都比较费钱或者费资源,所以在产品层面也希望我们对用户的能力使用次数做一定的限制,因此风控是必须的!2.为什么要自己写风控?那么多开源的风控组件,为什么还要写呢?是不是想…

一个大型网站架构的演变历程

正序&#xff1a; Rome was not built in a day&#xff08;罗马不是一天建成的。&#xff09;一个成熟的大型网站从来都不是一蹴而就的&#xff0c;需要经过多次架构的调整和升级&#xff0c;我们熟知的大型网站比如京东、淘宝、亚马逊&#xff0c;它们每天都有巨大的用户访问…

什么蓝牙耳机打电话效果最好?通话效果好的无线蓝牙耳机

2023年了&#xff0c;TWS耳机虽说近乎人手一只了&#xff0c;但用户换新的需求和呼声依然热火朝天&#xff0c;因为我们想要听音乐、刷视频的时候都得准备&#xff0c;下面整理一些通话效果不错的耳机品牌。 第一款&#xff1a;南卡小音舱蓝牙耳机 动圈单元&#xff1a;13.3m…

华为OD机试题,用 Java 解【靠谱的车】问题

最近更新的博客 华为OD机试题,用 Java 解【停车场车辆统计】问题华为OD机试题,用 Java 解【字符串变换最小字符串】问题华为OD机试题,用 Java 解【计算最大乘积】问题华为OD机试题,用 Java 解【DNA 序列】问题华为OD机试 - 组成最大数(Java) | 机试题算法思路 【2023】使…

基于springboot的微信公众号管理系统(支持多公众号接入)

微信公众号管理系统&#xff0c;支持多公众号接入。提供公众号菜单、自动回复、公众号素材、模板消息、CMS等管理功能 项目说明 是一个轻量级的公众号开发种子项目&#xff0c;可快速接入微信公众号管理功能swagger文档&#xff08;启动wx-api后查看&#xff09;&#xff1a;…

四信⾼速动态称重治超系统 不停车超载预检

随着交通运输行业的飞速发展&#xff0c;违法超载现象屡见不鲜&#xff0c;对公路、桥梁等造成了严重破坏&#xff0c;且容易引发交通事故。因此&#xff0c;有必要采用超载治理模式&#xff0c;有效延伸超限检测站管理上的时空范围、缓解执法力量不足的矛盾&#xff0c;以便进…

CAD坐标有哪些输入方式?来看看这些CAD坐标输入方式!

在CAD设计过程中&#xff0c;有时需要通过已知坐标点来画图&#xff0c;有时又需要通过已知角度和距离来画图&#xff0c;在这种情况下&#xff0c;由于已知条件不同&#xff0c;所以便需要用不同的方式来定位点。那么&#xff0c;你知道CAD坐标有哪些输入方式吗&#xff1f;本…

JUC(二)

1.可重入锁–ReentrantLock原理 1.1.非公平锁的实现原理 1.1.1.加锁解锁流程 1>.先从构造器开始看,默认为非公平锁,可以在构造函数中设置参数指定公平锁 public ReentrantLock() {sync = new NonfairSync(); }public ReentrantLock

Packet Tracer配置CHAP双向认证

Packet Tracer配置CHAP双向认证 【拓扑图】 【设备参数表】 设备 接口 IP地址 子网掩码 默认网关 R1 S0/0/0 192.168.12.1 255.255.255.0 N/A R2 S0/0/0 192.168.12.2 255.255.255.0 N/A 【实验步骤】 Router>en Router#conf Configuring from terminal, me…

论文笔记:A Time Series is Worth 64 Words: Long-term Forecasting with Transformers

ICLR 2023 比较简单&#xff0c;就不分intro、model这些了 1 核心思想1&#xff1a;patching 给定每个时间段的长度、划分的stride&#xff0c;将时间序列分成若干个时间段 时间段之间可以有重叠&#xff0c;也可以没有每一个时间段视为一个token 1.1 使用patching的好处 降…

Java常见启动命令 -jar、-server、-cp比较

文章目录Java程序常见启动方式java -jarjava -server与-client参数java -cpJava程序常见启动方式 当前java程序启动主要以 -jar、-server、-cp等几个命令启动 jar 程序&#xff0c;其中我们最常用的java -jar启动方式&#xff0c;通常我们需要将当前工程所依赖的所有包编译到一…

全网资料最全Java数据结构与算法(1)

一、数据结构和算法概述 1.1什么是数据结构&#xff1f; 官方解释&#xff1a; 数据结构是一门研究非数值计算的程序设计问题中的操作对象&#xff0c;以及他们之间的关系和操作等相关问题的学科。 大白话&#xff1a; 数据结构就是把数据元素按照一定的关系组织起来的集合&a…

leecode-C语言实现-28. 找出字符串中第一个匹配项的下标

一、题目给你两个字符串 haystack 和 needle &#xff0c;请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标&#xff08;下标从 0 开始&#xff09;。如果 needle 不是 haystack 的一部分&#xff0c;则返回 -1 。示例 1&#xff1a;输入&#xff1a;haystack …

flutter 微信聊天输入框

高仿微信聊天输入框&#xff0c;效果图如下&#xff08;目前都是静态展示&#xff0c;服务还没开始开发&#xff09;&#xff1a; 大家如果观察仔细的话 应该会发现&#xff0c;他输入框下面的高度 刚好就是 软键盘的高度&#xff1b;所以在这里就需要监听软键盘的高度。还要配…

zigbee与WIFI同频干扰问题

zigbee与WIFI同频干扰 为了降低Wifi信道与Zigbee信道的同频干扰问题&#xff0c;Zigbee联盟在《Zigbee Home Automation Public Application Profile》中推荐使用11,14,15,19,20,24,25这七个信道。 为什么呢&#xff0c;我们看一下Wifi和Zigbee的信道分布。 WiFi带宽对干扰的…