IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java发起GET请求的二三事 -> 正文阅读

[Java知识库]Java发起GET请求的二三事

一、拼接url

首先我们需要知道的是,url是要符合一定格式的,比如我们就不能在url中写“$”、“#”、中文、空格等。所以,我们这里采用application/x-www-form-urlencoded格式对请求参数进行编码,可参考如下代码:

    /**
     * 构造完整请求url(带参数)
     * @param url 请求url,形如:http://192.168.1.19
     * @param params 请求参数,key是参数名,value是参数值
     * @return 构造好的url,形如:http://192.168.1.19?name=mark&base=jinan
     */
    public static String buildUrl(String url, Map<String, String> params) throws UnsupportedEncodingException {

        StringBuilder urlBuilder = new StringBuilder(url);

        if (params != null) {
            Set<String> keySet = params.keySet();
            if (CollectionUtils.isNotEmpty(keySet)) {
                urlBuilder.append("?");

                for (String key : keySet) {
                    urlBuilder.append(key)
                            .append("=")
                            .append(URLEncoder.encode(params.get(key), StandardCharsets.UTF_8.name()))
                            .append("&");
                }

                urlBuilder.deleteCharAt(urlBuilder.length() - 1);
            }
        }

        return urlBuilder.toString();
    }

通过方法buildUrl,我们就完成了url的拼接工作。

二、对参数进行编码是什么意思?

“采用application/x-www-form-urlencoded格式对请求参数进行编码”,到底是什么意思呢?

首先我们可以先搞懂URLEncoder.encode(params.get(key), StandardCharsets.UTF_8.name())方法的第二个参数的作用是什么

String str = new String(charArrayWriter.toCharArray());
byte[] ba = str.getBytes(charset);

上面代码片段摘自URLEncoderencode方法,charArrayWriter.toCharArray()是待编码字符串的字符数组,charset就是我们传入的“utf-8”。可以看出来,这里首先是将待编码字符串使用“utf-8”进行编码。当然这里也可以根据具体的需求,采用其他的编码格式。

紧接着,再继续对经过“utf-8”编码后的字符串,采用application/x-www-form-urlencoded编码格式要求进行计算,相关算法源码如下:

String str = new String(charArrayWriter.toCharArray());
byte[] ba = str.getBytes(charset);
for (int j = 0; j < ba.length; j++) {
    out.append('%');
    char ch = Character.forDigit((ba[j] >> 4) & 0xF, 16);
    // converting to use uppercase letter as part of
    // the hex value if ch is a letter.
    if (Character.isLetter(ch)) {
        ch -= caseDiff;
    }
    out.append(ch);
    ch = Character.forDigit(ba[j] & 0xF, 16);
    if (Character.isLetter(ch)) {
        ch -= caseDiff;
    }
    out.append(ch);
}

总结一下,URLEncoder.encode(params.get(key), StandardCharsets.UTF_8.name())做了两件事:首先将待编码字符串使用“utf-8”进行编码;然后将编码结果再使用符合application/x-www-form-urlencoded编码格式要求进行计算。经过以上两步,就完成了对字符串的编码工作。

三、读取输出流

发起请求后,返回的结果是一个InputStream,使用下面的代码读取它:

    /**
     * 读取InputStream
     */
    private static String getContent(InputStream inputStream) throws IOException {
        try (BufferedReader bufferedReader = new BufferedReader(
                new InputStreamReader(inputStream, StandardCharsets.UTF_8.name()))) {

            String line;
            StringBuilder result = new StringBuilder();
            while ((line = bufferedReader.readLine()) != null) {
                result.append(line);
            }

            return result.toString();
        }
    }

注意,对于BufferedReaderInputStreamReaderInputStream三个流,只需要关闭最外层的即可,流会一层一层的向内关闭,我们可以看下BufferedReaderclose方法源码:

public void close() throws IOException {
    synchronized (lock) {
        if (in == null)
            return;
        try {
            in.close();
        } finally {
            in = null;
            cb = null;
        }
    }
}

这里的in变量,实际上就对应咱们代码里的InputStreamReader,然后InputStreamReader调用close方法时,会继续向内关闭。

那么您可能也要问了,getContent方法有关闭任何流的代码么,为什么我没看见?其实是这样的,因为BufferedReader实现了Closeable接口,那么只需要把BufferedReader方法写到try代码块里,java就会自动帮助我们关闭的。

四、使用java原生方式发起get请求

java原生api支持发起get请求,示例代码如下:

/**
 * 发起GET请求(使用java原生方法)
 * @param url 请求路径
 * @param params 请求参数
 * @param headers 请求头
 */
public static String doGet(String url, Map<String, String> params, Map<String, String> headers) throws Exception {
    // 1.构造完整请求url(带参数)
    String fullUrl = buildUrl(url, params);
    log.info("full url is: " + fullUrl);

    // 2.打开连接
    HttpURLConnection conn = (HttpURLConnection) new URL(fullUrl).openConnection();

    // 3.设置请求方法为get
    conn.setRequestMethod("GET");

    // 4.设置请求头
    if (headers != null) {
        Set<String> headerKeys = headers.keySet();
        if (CollectionUtils.isNotEmpty(headerKeys)) {
            for (String headerKey : headerKeys) {
                conn.setRequestProperty(headerKey, headers.get(headerKey));
            }
        }
    }

    // 5.获取响应状态码(非2开头状态码,认定请求失败)
    int statusCode = conn.getResponseCode();
    if (statusCode < HttpStatus.SC_OK || statusCode >= HttpStatus.SC_MULTIPLE_CHOICES) {
        log.error("status code is: " + statusCode);
        log.error("error message is: " + getContent(conn.getErrorStream()));
        throw new Exception("response status code wrong, status code is: " + statusCode);
    }

    // 6.获取请求结果
    return getContent(conn.getInputStream());
}

五、使用apache的HttpClient发起get请求

apache也提供了工具包,示例代码如下:

/**
  * 发起GET请求(使用apache的HttpClient)
  * @param url 请求路径
  * @param params 请求参数
  * @param headers 请求头
  */
public static String doGetByHttpClient(String url, Map<String, String> params, Map<String, String> headers) throws Exception {
    try (CloseableHttpClient client = HttpClients.createDefault()) {
        // 1.构造完整请求url(带参数)
        String fullUrl = buildUrl(url, params);
        log.info("full url is: " + fullUrl);

        // 2.创建httpGet
        HttpGet httpGet = new HttpGet(fullUrl);

        // 3.设置请求头
        if (headers != null) {
            Set<String> headerKeys = headers.keySet();
            if (CollectionUtils.isNotEmpty(headerKeys)) {
                for (String headerKey : headerKeys) {
                    httpGet.setHeader(headerKey, headers.get(headerKey));
                }
            }
        }

        // 4.发起请求
        CloseableHttpResponse response = client.execute(httpGet);

        // 5.获取响应状态码(非2开头状态码,认定请求失败)
        StatusLine statusLine = response.getStatusLine();
        int statusCode = statusLine.getStatusCode();
        if (statusCode < HttpStatus.SC_OK || statusCode >= HttpStatus.SC_MULTIPLE_CHOICES) {
            log.error("status code is: " + statusCode);
            log.error("error message is: " + statusLine.getReasonPhrase());
            throw new Exception("response status code wrong, status code is: " + statusCode);
        }

        // 6.获取请求结果
        return getContent(response.getEntity().getContent());
    }
}

就我写以上代码的经验来看,使用原生的或者httpClient,差别不大,我个人反而更喜欢原生的。各位大佬看自己喜好吧。

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-06-08 18:52:54  更:2022-06-08 18:57:00 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 19:21:08-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码