HttpClient 4.5.x 工具类设计与实现
最近,业务需要在java服务端发起http请求,需要实现"GET","POST","PUT"等基本方法。于是想以 "HttpClient" 为基础,封装基本实现方法。在github上面看到一个项目cn-umbrella/httpclientDemo,里面实现简单的 "GET","POST"。一方面,demo中实现的逻辑不能满足业务需要,同时实现过程中,存在使用过期的类,代码封装不足等问题。也是我就重新封装了httpclient-util,抱着大家相互学习的心态,将代码上传到github上面,相关链接,错误的地方欢迎大家指正。
回归正题,下面讲讲工具类实现的基本思路。
1.主要依赖
<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpclient</artifactId>
<version>4.5.2</version>
</dependency>
<!--httpmime主要用于构建multi-part/form-data 请求-->
<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpmime</artifactId>
<version>4.5.2</version>
</dependency>
2.HttpClientUtil 主要方法
/**
* Do http request
* @param request request object
* @return response object
*/
public static Response doRequest(Request request)
其中,Request 封装了请求参数,Response 封装了返回内容。
3.Request 说明
支持方法:"GET","OPTIONS","HEAD","POST","PUT","PATCH"。支持请求体: "application/x-www-form-urlencoded","json字符串","multipart/form-data"。
Request的继承关系
Request (没有请求body的请求,支持方法:"GET", "OPTIONS", "HEAD")
|---BaseEntityRequest (抽象类,定义有请求body的请求, 支持方法:"POST", "PUT", "PATCH")
|--- UrlEncodedFormRequest (application/x-www-form-urlencoded)
|--- JsonRequest (请求body为json字符串)
|--- MultiPartFormRequest (multipart/form-data)
同时,Request 支持使用 ssl/tls,使用request.setUseSSL(true);设置即可。
HttpEntity就是对应请求body相关实体对象。
3.BaseEntityRequest 介绍
任何实现具有请求body的请求,需要实现 BaseEntityRequest 以下方法
/**
* Get HttpEntity about request body
* @return HttpEntity
*/
public abstract HttpEntity getEntity();
3.1 UrlEncodedFormRequest 核心代码
这个实现比较简单,"K-V" 的形式,返回 UrlEncodedFormEntity,注意参数编码。
@Override
public HttpEntity getEntity() {
List<NameValuePair> pairList = new ArrayList<>(params.size());
for (Map.Entry<String, Object> entry : params.entrySet()) {
NameValuePair pair = new BasicNameValuePair(entry.getKey(), entry
.getValue().toString());
pairList.add(pair);
}
return new UrlEncodedFormEntity(pairList, Charset.forName(getRequestCharset())); //参数编码
}
3.2 JsonRequest 核心代码
请求body就是一个 "json字符串",返回一个StringEntity 实体即可。
@Override
public HttpEntity getEntity() {
StringEntity stringEntity = new StringEntity(jsonObject.toString(), getRequestCharset()); //解决中文乱码问题,需要注意编码,如使用"utf-8"
stringEntity.setContentEncoding(getResponseDefaultCharset());
stringEntity.setContentType(JSON_CONTENT_TYPE);
return stringEntity;
}
3.3 MultiPartFormRequest 核心代码
httpclient 类库中不直接支持构造 "multi-part/form-data" ,需要引入 httpmime,并使用 MultipartEntityBuilder 构造请求body。如果是文件类型的参数对象,使用 FileBody 构建 part,否则,使用 StringBody,注意编码。
@Override
public HttpEntity getEntity() {
try {
Charset charset = CharsetUtils.get(getRequestCharset());
//使用浏览器兼容模式,并且设置编码,防止文件名乱码问题
MultipartEntityBuilder builder = MultipartEntityBuilder.create()
.setMode(HttpMultipartMode.BROWSER_COMPATIBLE)
.setCharset(charset); //编码
parts.forEach((key, value) -> {
if (null != value && value instanceof File) { //for file
FileBody fileBody = new FileBody((File) value);
builder.addPart(key, fileBody);
} else {
StringBody stringBody = new StringBody(null == value ? "" : value.toString()
, ContentType.create(ContentType.TEXT_PLAIN.getMimeType(), charset)); //编码
builder.addPart(key, stringBody);
}
});
return builder.build();
} catch (UnsupportedEncodingException e) {
throw new RuntimeException("Charset '" + getRequestCharset() + "' is unsupported!");
}
}
4.注意地方
使用 https 过程中,如果出现下述异常:
javax.net.ssl.SSLHandshakeException: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target
at sun.security.ssl.Alerts.getSSLException(Alerts.java:192)
at sun.security.ssl.SSLSocketImpl.fatal(SSLSocketImpl.java:1949)
at sun.security.ssl.Handshaker.fatalSE(Handshaker.java:302)
at sun.security.ssl.Handshaker.fatalSE(Handshaker.java:296)
...
Caused by: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target
at sun.security.validator.PKIXValidator.doBuild(PKIXValidator.java:387)
at sun.security.validator.PKIXValidator.engineValidate(PKIXValidator.java:292)
at sun.security.validator.Validator.validate(Validator.java:260)
at sun.security.ssl.X509TrustManagerImpl.validate(X509TrustManagerImpl.java:324)
at sun.security.ssl.X509TrustManagerImpl.checkTrusted(X509TrustManagerImpl.java:229)
可以参考我的上篇博文解决 java 使用ssl过程中出现"PKIX path building failed:...,当然在包 com.xiaobenma020.http.cert 下面可以找到类 InstallCert
关键步骤
% java InstallCert _web_site_hostname_
显示相关的证书信息
此时输入'q' 则为'退出', '1' 则将添加CA证书。
将新生成的 "jssecacerts" 移到"$JAVA_HOME/jre/lib/security"
5.使用demo
@Test
public void doResponse() throws Exception {
UrlEncodedFormRequest request = new UrlEncodedFormRequest("https://xxx.com/login", RequestMethod.POST);
//url form param
request.addParam("loginId", "loginId");
request.addParam("password", "password");
//query string param
request.addUrlParam("version", "v1");
//ssl
request.setUseSSL(true);
Response response = HttpClientUtil.doRequest(request);
System.out.println(response.getResponseText()); //response text
System.out.println(response.getCode()); //response code
System.out.println(response.getHeader("Set-Cookie"));
}
HttpClient 4.5.x 工具类设计与实现的更多相关文章
- JFreeChart绘制XY折线图(工具类设计)
准备用Java写通信的仿真平台作为毕业设计,相比matlab绘图,Java绘图需要自己去写很多工具类,博主在这采用了JFreeChart的开源解决方案,摸索着自己写了一个XY折线图工具类,话不多说贴源 ...
- httpclient 实现的http工具类
HttpClient实现的工具类 就是簡單的用http 協議請求請求地址並返回數據,廢話少數直接上代碼 http請求返回的封裝類 package com.nnk.upstream.util; impo ...
- 使用RedisTemplate进行Redis存取的工具类设计
通常在访问量大数据更新频率不高的系统中会使用第三方的缓存组件来降低数据库服务的负载,鉴于模块独立分工独立的考虑,针对缓存组件操作的工作全部应该统一接口对其他业务提供服务,这样业务操作只需要关注业务实现 ...
- 基于Java反射的map自动装配JavaBean工具类设计
我们平时在用Myabtis时不是常常需要用map来传递参数,大体是如下的步骤: public List<Role> findRoles(Map<String,Object> p ...
- 使用HttpClient访问url的工具类
maven依赖jar包配置: <dependency> <groupId>org.apache.httpcomponents</groupId> <artif ...
- Httpclient 工具类(get,put)
package com.googosoft.until; import java.io.IOException; import org.apache.http.HttpEntity; import o ...
- HttpUtil工具类,发送Get/Post请求,支持Http和Https协议
HttpUtil工具类,发送Get/Post请求,支持Http和Https协议 使用用Httpclient封装的HttpUtil工具类,发送Get/Post请求 1. maven引入httpclien ...
- Java基础进阶:APi使用,Math,Arrarys,Objects工具类,自动拆装箱,字符串与基本数据类型互转,递归算法源码,冒泡排序源码实现,快排实现源码,附重难点,代码实现源码,课堂笔记,课后扩展及答案
要点摘要 Math: 类中么有构造方法,内部方法是静态的,可以直接类名.方式调用 常用: Math.abs(int a):返回参数绝对值 Math.ceil(double a):返回大于或等于参数的最 ...
- 170314、工具:apache httpClient多线程并发情况下安全实用及工具类分享
简单用法介绍:介绍来源网络 建立连接:在HttpClient中使用多线程的一个主要原因是可以一次执行多个方法.在执行期间,每一个方法都使用一个HttpConnection实例.由于在同一时间多个连接只 ...
随机推荐
- axis2打包方式发布
参照http://gao-xianglong.iteye.com/blog/1744557这篇文章,注意的是打包services.xml的时候要将它的上级目录meta-inf一起打包,放到axis2\ ...
- kali 渗透的一些笔记
kali实战笔记 17:55 2016/7/19 by: _Silvers kali系统安装后的配置及美化安装vmwareToolstar zxvf VMwareTools-sfsfsfasfasfs ...
- 夺命雷公狗-----React---21--小案例之心情留言板
这个功能如果是用传统型的jquery来写都要花费很多时间才可以完成的案例, 亲测jquery配合bootstrap来写和bootstrap配合react.js来写,不知不觉中有点震惊... jquer ...
- 使用百度地图api接口获取公交地图路线和车站
需要在页面文件中引用百度的js @*<script type="text/javascript" src="http://api.map.baidu.com/api ...
- 读取Devexpress内部的图标
1.图标在Dev源码的存储路径: Sources D.x.u 15.1.3\DevExpress.Images\Images 2.引用DevExpress.Images.v15.1.dll文件,代 ...
- Xcode 如何删除过期的Provisioning Profile文件
Xcode 中所有的Provisioning Profile文件,都在 ~/Library/MobileDevice/Provisioning Profiles 这个文件夹下:进入该文件夹,按照文件 ...
- 剑指offer——树的子结构 (JAVA代码)
版权声明:本文为博主原创文章,未经博主允许不得转载. 题目描述: 输入两棵二叉树A,B,判断B是不是A的子结构.(ps:我们约定空树不是任意一个树的子结构). 解题思路: 首先看牛客网给出的测试用例: ...
- php正则表达式 常用记录
一正则汉字匹配数字id 汉字 $str = '<a href="http://music.baidu.com/song/121353608" target="_bl ...
- int型、long型和long long型
long long本质上还是整型,只不过是一种超长的整型.int型:32位整型,取值范围为-2^31 ~ (2^31 - 1) . long:在32位系统是32位整型,取值范围为-2^31 ~ (2^ ...
- centos 更换 yum源
阿里云Linux安装镜像源地址:http://mirrors.aliyun.com/ CentOS系统更换软件安装源第一步:备份你的原镜像文件,以免出错后可以恢复. mv /etc/yum.repos ...