`

httpClient 研究

 
阅读更多
虽然JDK 的 java.net 包中已经提供了访问 HTTP 协议的基本功能,但是对于大部分应用程序来说,JDK 库本身提供的功能还不够丰富和灵活。

      HttpClient 是 Apache Jakarta Common 下的子项目,用来提供高效的、最新的、功能丰富的支持 HTTP 协议的客户端编程工具包。

Http协议的重要性相信不用我多说了,HttpClient相比传统JDK自带的URLConnection,增加了易用性和灵活性(具体区别,日后我们再讨论),它不仅是客户端发送Http请求变得容易,而且也方便了开发人员测试接口(基于Http协议的),即提高了开发的效率,也方便提高代码的健壮性。因此熟练掌握HttpClient是很重要的必修内容,掌握HttpClient后,相信对于Http协议的了解会更加深入。
一、简介
HttpClient是Apache Jakarta Common下的子项目,用来提供高效的、最新的、功能丰富的支持HTTP协议的客户端编程工具包,并且它支持HTTP协议最新的版本和建议。HttpClient已经应用在很多的项目中,比如Apache Jakarta上很著名的另外两个开源项目Cactus和HTMLUnit都使用了HttpClient。
下载地址: http://hc.apache.org/downloads.cgi
二、特性
1. 基于标准、纯净的java语言。实现了Http1.0和Http1.1
2. 以可扩展的面向对象的结构实现了Http全部的方法(GET, POST, PUT, DELETE, HEAD, OPTIONS, and TRACE)。
3. 支持HTTPS协议。
4. 通过Http代理建立透明的连接。
5. 利用CONNECT方法通过Http代理建立隧道的https连接。
6. Basic, Digest, NTLMv1, NTLMv2, NTLM2 Session, SNPNEGO/Kerberos认证方案。
7. 插件式的自定义认证方案。
8. 便携可靠的套接字工厂使它更容易的使用第三方解决方案。
9. 连接管理器支持多线程应用。支持设置最大连接数,同时支持设置每个主机的最大连接数,发现并关闭过期的连接。
10. 自动处理Set-Cookie中的Cookie。
11. 插件式的自定义Cookie策略。
12. Request的输出流可以避免流中内容直接缓冲到socket服务器。
13. Response的输入流可以有效的从socket服务器直接读取相应内容。
14. 在http1.0和http1.1中利用KeepAlive保持持久连接。
15. 直接获取服务器发送的response code和 headers。
16. 设置连接超时的能力。
17. 实验性的支持http1.1 response caching。
18. 源代码基于Apache License 可免费获取。
三、使用方法
使用HttpClient发送请求、接收响应很简单,一般需要如下几步即可。
1. 创建HttpClient对象。
2. 创建请求方法的实例,并指定请求URL。如果需要发送GET请求,创建HttpGet对象;如果需要发送POST请求,创建HttpPost对象。
3. 如果需要发送请求参数,可调用HttpGet、HttpPost共同的setParams(HetpParams params)方法来添加请求参数;对于HttpPost对象而言,也可调用setEntity(HttpEntity entity)方法来设置请求参数。
4. 调用HttpClient对象的execute(HttpUriRequest request)发送请求,该方法返回一个HttpResponse。
5. 调用HttpResponse的getAllHeaders()、getHeaders(String name)等方法可获取服务器的响应头;调用HttpResponse的getEntity()方法可获取HttpEntity对象,该对象包装了服务器的响应内容。程序可通过该对象获取服务器的响应内容。
6. 释放连接。无论执行方法是否成功,都必须释放连接
四、实例
[java] view plaincopy在CODE上查看代码片派生到我的代码片
package com.test; 
 
import java.io.File; 
import java.io.FileInputStream; 
import java.io.IOException; 
import java.io.UnsupportedEncodingException; 
import java.security.KeyManagementException; 
import java.security.KeyStore; 
import java.security.KeyStoreException; 
import java.security.NoSuchAlgorithmException; 
import java.security.cert.CertificateException; 
import java.util.ArrayList; 
import java.util.List; 
 
import javax.net.ssl.SSLContext; 
 
import org.apache.http.HttpEntity; 
import org.apache.http.NameValuePair; 
import org.apache.http.ParseException; 
import org.apache.http.client.ClientProtocolException; 
import org.apache.http.client.entity.UrlEncodedFormEntity; 
import org.apache.http.client.methods.CloseableHttpResponse; 
import org.apache.http.client.methods.HttpGet; 
import org.apache.http.client.methods.HttpPost; 
import org.apache.http.conn.ssl.SSLConnectionSocketFactory; 
import org.apache.http.conn.ssl.SSLContexts; 
import org.apache.http.conn.ssl.TrustSelfSignedStrategy; 
import org.apache.http.entity.ContentType; 
import org.apache.http.entity.mime.MultipartEntityBuilder; 
import org.apache.http.entity.mime.content.FileBody; 
import org.apache.http.entity.mime.content.StringBody; 
import org.apache.http.impl.client.CloseableHttpClient; 
import org.apache.http.impl.client.HttpClients; 
import org.apache.http.message.BasicNameValuePair; 
import org.apache.http.util.EntityUtils; 
import org.junit.Test; 
 
public class HttpClientTest { 
 
    @Test 
    public void jUnitTest() { 
        get(); 
    } 
 
    /**
     * HttpClient连接SSL
     */ 
    public void ssl() { 
        CloseableHttpClient httpclient = null; 
        try { 
            KeyStore trustStore = KeyStore.getInstance(KeyStore.getDefaultType()); 
            FileInputStream instream = new FileInputStream(new File("d:\\tomcat.keystore")); 
            try { 
                // 加载keyStore d:\\tomcat.keystore   
                trustStore.load(instream, "123456".toCharArray()); 
            } catch (CertificateException e) { 
                e.printStackTrace(); 
            } finally { 
                try { 
                    instream.close(); 
                } catch (Exception ignore) { 
                } 
            } 
            // 相信自己的CA和所有自签名的证书 
            SSLContext sslcontext = SSLContexts.custom().loadTrustMaterial(trustStore, new TrustSelfSignedStrategy()).build(); 
            // 只允许使用TLSv1协议 
            SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslcontext, new String[] { "TLSv1" }, null, 
                    SSLConnectionSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); 
            httpclient = HttpClients.custom().setSSLSocketFactory(sslsf).build(); 
            // 创建http请求(get方式) 
            HttpGet httpget = new HttpGet("https://localhost:8443/myDemo/Ajax/serivceJ.action"); 
            System.out.println("executing request" + httpget.getRequestLine()); 
            CloseableHttpResponse response = httpclient.execute(httpget); 
            try { 
                HttpEntity entity = response.getEntity(); 
                System.out.println("----------------------------------------"); 
                System.out.println(response.getStatusLine()); 
                if (entity != null) { 
                    System.out.println("Response content length: " + entity.getContentLength()); 
                    System.out.println(EntityUtils.toString(entity)); 
                    EntityUtils.consume(entity); 
                } 
            } finally { 
                response.close(); 
            } 
        } catch (ParseException e) { 
            e.printStackTrace(); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        } catch (KeyManagementException e) { 
            e.printStackTrace(); 
        } catch (NoSuchAlgorithmException e) { 
            e.printStackTrace(); 
        } catch (KeyStoreException e) { 
            e.printStackTrace(); 
        } finally { 
            if (httpclient != null) { 
                try { 
                    httpclient.close(); 
                } catch (IOException e) { 
                    e.printStackTrace(); 
                } 
            } 
        } 
    } 
 
    /**
     * post方式提交表单(模拟用户登录请求)
     */ 
    public void postForm() { 
        // 创建默认的httpClient实例.   
        CloseableHttpClient httpclient = HttpClients.createDefault(); 
        // 创建httppost   
        HttpPost httppost = new HttpPost("http://localhost:8080/myDemo/Ajax/serivceJ.action"); 
        // 创建参数队列   
        List<namevaluepair> formparams = new ArrayList<namevaluepair>(); 
        formparams.add(new BasicNameValuePair("username", "admin")); 
        formparams.add(new BasicNameValuePair("password", "123456")); 
        UrlEncodedFormEntity uefEntity; 
        try { 
            uefEntity = new UrlEncodedFormEntity(formparams, "UTF-8"); 
            httppost.setEntity(uefEntity); 
            System.out.println("executing request " + httppost.getURI()); 
            CloseableHttpResponse response = httpclient.execute(httppost); 
            try { 
                HttpEntity entity = response.getEntity(); 
                if (entity != null) { 
                    System.out.println("--------------------------------------"); 
                    System.out.println("Response content: " + EntityUtils.toString(entity, "UTF-8")); 
                    System.out.println("--------------------------------------"); 
                } 
            } finally { 
                response.close(); 
            } 
        } catch (ClientProtocolException e) { 
            e.printStackTrace(); 
        } catch (UnsupportedEncodingException e1) { 
            e1.printStackTrace(); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        } finally { 
            // 关闭连接,释放资源   
            try { 
                httpclient.close(); 
            } catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 
    } 
 
    /**
     * 发送 post请求访问本地应用并根据传递参数不同返回不同结果
     */ 
    public void post() { 
        // 创建默认的httpClient实例.   
        CloseableHttpClient httpclient = HttpClients.createDefault(); 
        // 创建httppost   
        HttpPost httppost = new HttpPost("http://localhost:8080/myDemo/Ajax/serivceJ.action"); 
        // 创建参数队列   
        List<namevaluepair> formparams = new ArrayList<namevaluepair>(); 
        formparams.add(new BasicNameValuePair("type", "house")); 
        UrlEncodedFormEntity uefEntity; 
        try { 
            uefEntity = new UrlEncodedFormEntity(formparams, "UTF-8"); 
            httppost.setEntity(uefEntity); 
            System.out.println("executing request " + httppost.getURI()); 
            CloseableHttpResponse response = httpclient.execute(httppost); 
            try { 
                HttpEntity entity = response.getEntity(); 
                if (entity != null) { 
                    System.out.println("--------------------------------------"); 
                    System.out.println("Response content: " + EntityUtils.toString(entity, "UTF-8")); 
                    System.out.println("--------------------------------------"); 
                } 
            } finally { 
                response.close(); 
            } 
        } catch (ClientProtocolException e) { 
            e.printStackTrace(); 
        } catch (UnsupportedEncodingException e1) { 
            e1.printStackTrace(); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        } finally { 
            // 关闭连接,释放资源   
            try { 
                httpclient.close(); 
            } catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 
    } 
 
    /**
     * 发送 get请求
     */ 
    public void get() { 
        CloseableHttpClient httpclient = HttpClients.createDefault(); 
        try { 
            // 创建httpget.   
            HttpGet httpget = new HttpGet("http://www.baidu.com/"); 
            System.out.println("executing request " + httpget.getURI()); 
            // 执行get请求.   
            CloseableHttpResponse response = httpclient.execute(httpget); 
            try { 
                // 获取响应实体   
                HttpEntity entity = response.getEntity(); 
                System.out.println("--------------------------------------"); 
                // 打印响应状态   
                System.out.println(response.getStatusLine()); 
                if (entity != null) { 
                    // 打印响应内容长度   
                    System.out.println("Response content length: " + entity.getContentLength()); 
                    // 打印响应内容   
                    System.out.println("Response content: " + EntityUtils.toString(entity)); 
                } 
                System.out.println("------------------------------------"); 
            } finally { 
                response.close(); 
            } 
        } catch (ClientProtocolException e) { 
            e.printStackTrace(); 
        } catch (ParseException e) { 
            e.printStackTrace(); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        } finally { 
            // 关闭连接,释放资源   
            try { 
                httpclient.close(); 
            } catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 
    } 
 
    /**
     * 上传文件
     */ 
    public void upload() { 
        CloseableHttpClient httpclient = HttpClients.createDefault(); 
        try { 
            HttpPost httppost = new HttpPost("http://localhost:8080/myDemo/Ajax/serivceFile.action"); 
 
            FileBody bin = new FileBody(new File("F:\\image\\sendpix0.jpg")); 
            StringBody comment = new StringBody("A binary file of some kind", ContentType.TEXT_PLAIN); 
 
            HttpEntity reqEntity = MultipartEntityBuilder.create().addPart("bin", bin).addPart("comment", comment).build(); 
 
            httppost.setEntity(reqEntity); 
 
            System.out.println("executing request " + httppost.getRequestLine()); 
            CloseableHttpResponse response = httpclient.execute(httppost); 
            try { 
                System.out.println("----------------------------------------"); 
                System.out.println(response.getStatusLine()); 
                HttpEntity resEntity = response.getEntity(); 
                if (resEntity != null) { 
                    System.out.println("Response content length: " + resEntity.getContentLength()); 
                } 
                EntityUtils.consume(resEntity); 
            } finally { 
                response.close(); 
            } 
        } catch (ClientProtocolException e) { 
            e.printStackTrace(); 
        } catch (IOException e) { 
            e.printStackTrace(); 
        } finally { 
            try { 
                httpclient.close(); 
            } catch (IOException e) { 
                e.printStackTrace(); 
            } 
        } 
    } 
}</namevaluepair></namevaluepair></namevaluepair></namevaluepair> 
本实例是采用HttpClient4.3最新版本。该版本与之前的代码写法风格相差较大,大家多留意下。
以上内容来自http://blog.csdn.net/wangpeng047/article/details/19624529

对于https的请求,有的网站在没有证书的情况下可以通过,
HttpClient 如何忽略证书验证 - ALLOW_ALL_HOSTNAME_VERIFIER
设置可以访问HTTPS,但有的网站则必须要有证书。


    1、证书相关问题。
生成keystore: {jkd-path}/bin中有一个keytool,通过命令生成xxx.keystrore

keytool -genkey  -keyalg RSA -validity 2000 -keystore xxx.keystrore
按照提示完成生成过程:


现在只是生成了一个存放证书的"库",我们需要把访问的server端的证书导入到我们刚才创建的"库"中。
(具体每个浏览器导出证书的过程类似)
Chrome 导出证书过程:

具体导出的cer方式应该都是可以的,我是默认导出第一个格式的。这里需要注意的是,导出的这个证书域名很重要,这个要与你代码里的域名保持一致,否则会提示域名不匹配导致请求被拒绝。

现在我们需要将导出的证书导入到我们刚才生成的xxx.keystore中,命令如下

keytool -importcert -trustcacerts -file zzz.cer -kestore xxx.keystore
其中-zzz.cer 就是你刚才导出证书的文件名。
其中-xxx.keystore就是你刚才生成的"库"。
    2、java代码实现

/**
*
*/
package demo.test;

import java.io.IOException;
import java.io.InputStream;
import java.security.KeyManagementException;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.UnrecoverableKeyException;
import java.security.cert.CertificateException;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.conn.scheme.Scheme;
import org.apache.http.conn.ssl.SSLSocketFactory;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;

/**
* @author Hero_Q
*
*/
public class HTTPSDemo {

    private static final String SERVER = "https://xxx.com";

    public static void main(String[] args) {
        HttpClient httpClient = new DefaultHttpClient();
        try {
            KeyStore trustStore = KeyStore.getInstance(KeyStore.getDefaultType());
            InputStream in = HTTPSDemo.class.getResourceAsStream("xxx.keystore");
           //自己生成的库的密码,
trustStore.load(in, "123456".toCharArray());
            in.close();
            SSLSocketFactory socketFactory = new SSLSocketFactory(trustStore);
            Scheme sch = new Scheme("https", 443, socketFactory);
            httpClient.getConnectionManager().getSchemeRegistry().register(sch);

            HttpPost httpPost = new HttpPost(SERVER);
            System.out.println("executing request" + httpPost.getRequestLine());
            // 执行请求
            HttpResponse response = httpClient.execute(httpPost);// 获得响应实体
            HttpEntity entity = response.getEntity();

            System.out.println("----------------------------------------");
            System.out.println(response.getStatusLine());
            System.out.println(entity);
            if (entity != null) {
                System.out.println("Response content length: " + entity.getContentLength());
            }
            // 销毁实体
            EntityUtils.consume(entity);
        } catch (IOException | KeyStoreException | NoSuchAlgorithmException | CertificateException | KeyManagementException | UnrecoverableKeyException e) {
            e.printStackTrace();
        } finally {
            // 当不再需要HttpClient实例时,关闭连接管理器以确保释放所有占用的系统资源
            httpClient.getConnectionManager().shutdown();
        }
    }

}
(我用的是java7),代码中需要指定的文件路径就是刚才生成的xxx.keystore库,具体原理就是构造请求,注册https协议,加载keystore,发送请求~我本地测试是可以通过的,具体的post请求中还需要加入具体的请求参数。不过最基本的还是http、https这些底层的东西应该清晰,要不仅仅的是把别人的代码copy下,跑通了,再次遇到类似的问题或许优化时会很无奈... 并且httpclient 发布的4.3的api还是有些改动的,如果不知道最基本的原理,永远只能停留在copy的阶段.... 加油,共同进步~。
分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    HTTPClient 的一个封装

    解压并研究这些文件,可以帮助我们理解具体是如何进行封装的,也可以为我们的项目提供参考和学习的素材。 总结来说,HttpClient的封装是为了提高代码的可维护性和易用性,通过隐藏复杂实现细节,提供简洁的API接口...

    Apache httpclient源码4.5.12

    Apache HttpClient 是一个强大的Java库,用于执行HTTP请求。... 首先,HttpClient的`HttpClient`类是整个库的核心,它是发送HTTP请求的...通过研究源码,你可以更好地理解和定制HttpClient的行为,以满足特定项目的需求。

    apache httpclient 源码和 jar包

    研究源码可以帮助我们理解如何在高并发环境下正确地使用HttpClient。 4. **错误处理**:通过源码,我们可以看到HttpClient如何处理网络异常、超时等问题,这对于我们在实际开发中遇到类似问题时提供解决方案有很大的...

    HttpClient 4.4.1 码源文件src

    在这个版本中,我们可以深入研究其源代码来了解其内部工作原理和优化我们的HTTP请求。 HttpClient 4.4.1的核心组件包括以下几个方面: 1. **HttpClient核心模块**:这部分包含了执行HTTP请求和响应的主要类,如`...

    httpclient 4.0.3 源代码

    《HttpClient 4.0.3源代码解析》 HttpClient是一个由Apache基金会开发的开源HTTP客户端API,广泛应用于Java编程环境中...深入研究源代码,能帮助我们更好地应对各种网络环境和应用场景,提升应用程序的健壮性和效率。

    使用httpclient模拟表单提交

    标签"源码"意味着你可以进一步研究HttpClient的内部实现,了解它是如何处理网络请求的。至于"工具",HttpClient可以作为一个工具库被集成到各种项目中,以简化HTTP通信任务。 在实际应用中,可能还需要处理更复杂的...

    HttpClient4.5源码

    总的来说,HttpClient 4.5源码的研究对于任何希望在Java应用中进行HTTP通信的开发者来说都是一份宝贵的资源,它不仅可以帮助我们理解HTTP协议的实现细节,还能让我们学习到如何优雅地处理网络通信中的各种复杂情况。...

    HttpClient4.5.3资源

    HttpClient是Apache软件基金会的一个开源HTTP...通过深入研究源码,开发者可以了解其内部工作原理,进一步优化和定制自己的HTTP通信方案。同时,配合文档学习,可以更高效地掌握HttpClient的使用技巧,提升开发效率。

    commons-httpclient 源代码包

    通过阅读和研究源代码,开发者可以深入理解如何在Java中构建一个高效的HTTP客户端,并且可以借鉴其中的设计模式和最佳实践。同时,`commons-httpclient`也常被用作其他库的基础,比如在早期的Spring框架中,就曾经...

    爬数据案例 httpclient htmlparser

    在IT领域,网络爬虫是获取大量数据的重要手段,尤其对于数据分析、研究和自动化信息处理。本案例聚焦于使用Java语言结合`httpclient`库进行网络请求,以及`htmlparser`库解析HTML文档,实现一个基础的爬虫程序。下面...

    浏览器发送模拟httpclient软件

    当我们谈论“浏览器发送模拟httpclient软件”时,我们指的是一个能够模拟浏览器行为的应用程序,允许用户通过编程方式执行...无论是进行API接口测试,还是研究网络请求的工作流程,这类工具都是开发者不可或缺的助手。

    HttpClient for android 4 3 5 jar

    HttpClient是Apache基金会开发的一个HTTP客户端库,广泛应用于Java和Android平台。在Android中,HttpClient库提供了与Web服务器进行...通过研究源码,开发者可以更好地理解和利用这个库,解决在实际开发中遇到的问题。

    commons-httpclient-3.0.1-src.zip

    源码中,我们可以研究连接的创建、关闭、回收策略,以及超时和重试机制。 3. **请求和响应处理**:HttpClient 封装了请求头、请求体、响应头和响应体的处理逻辑。通过源码,我们可以学习如何设置和获取 HTTP 头部...

    httpclient

    这意味着我们可以详细研究HttpClient的API用法,查看如何初始化HttpClient对象,创建HttpGet或HttpPost实例,添加请求头,以及如何处理HttpEntity响应。 HttpClient的使用步骤大致如下: 1. **创建HttpClient实例*...

    springboot学习demo,springmvc,httpclient微服务架构之一

    尽管"hzrPayDemo"标题提到微服务架构,但没有明确指出具体使用了Spring Cloud的哪些组件,可能需要进一步研究项目源码来了解。 HTTPClient是Apache HttpClient库,它允许Java应用以HTTP协议与其他Web服务器进行交互...

    PB_Json_httpclient_crypto_ftp_20210125.rar

    标题 "PB_Json_httpclient_crypto_ftp_20210125.rar" 暗示了这个压缩包包含了与编程相关的组件和教程,主要...通过解压并研究这些文件,开发者可以深入理解如何在PB环境中实现这些功能,并提升他们的应用程序开发技能。

    轻量级C++实现的httpserver和httpclient

    在IT领域,C++是一种强大的、面向对象的编程语言,被广泛用于开发系统软件、游戏引擎、嵌入式系统以及高性能的应用程序。本项目聚焦于轻量级的C++...无论是为了学习还是实际项目开发,这都是一个值得研究和参考的实例。

    httpclient 4.3.6

    在 HttpClient 4.3.6 版本中,它不仅包含了最新的特性改进,还提供了完整的源代码供开发者深入研究和定制。 一、HttpClient 的核心概念与功能 HttpClient 主要提供了一系列接口和类,用于执行 HTTP 请求、处理响应...

    C++/CLI应用程序调用httpClient请求HTTP和用Newtonsoft解析json示例代码

    现在C++也不行了吗,网上找不到相关代码。为了搞清楚C++中如何使用httpClient和Newtonsoft,研究了一下午。我从未用C++编程,一直用C#,现学现卖。

    HttpClient使用

    对于更深入的理解,可以研究HttpClient的源码,了解其内部实现细节。例如,理解连接池的工作原理,或者查看如何处理HTTP状态码和异常。 ### 文件EdaLibDownloadForDxD 虽然这个文件名与HttpClient主题不直接相关,...

Global site tag (gtag.js) - Google Analytics