一、简介
HttpClient是Apache Jakarta Common下的子项目,用来提供高效的、最新的、功能丰富的支持HTTP协议的客户端编程工具包,并且它支持HTTP协议最新的版本和建议。HttpClient已经应用在很多的项目中,比如Apache Jakarta上很著名的另外两个开源项目Cactus和HTMLUnit都使用了HttpClient。
下载地址: http://hc.apache.org/downloads.cgi
二、特性
-
基于标准、纯净的java语言。实现了Http1.0和Http1.1
-
以可扩展的面向对象的结构实现了Http全部的方法(GET, POST, PUT, DELETE, HEAD, OPTIONS, and TRACE)。
-
支持HTTPS协议。
-
通过Http代理建立透明的连接。
-
利用CONNECT方法通过Http代理建立隧道的https连接。
-
Basic, Digest, NTLMv1, NTLMv2, NTLM2 Session, SNPNEGO/Kerberos认证方案。
-
插件式的自定义认证方案。
-
便携可靠的套接字工厂使它更容易的使用第三方解决方案。
-
连接管理器支持多线程应用。支持设置最大连接数,同时支持设置每个主机的最大连接数,发现并关闭过期的连接。
-
自动处理Set-Cookie中的Cookie。
-
插件式的自定义Cookie策略。
-
Request的输出流可以避免流中内容直接缓冲到socket服务器。
-
Response的输入流可以有效的从socket服务器直接读取相应内容。
-
在http1.0和http1.1中利用KeepAlive保持持久连接。
-
直接获取服务器发送的response code和 headers。
-
设置连接超时的能力。
-
实验性的支持http1.1 response caching。
-
源代码基于Apache License 可免费获取。
二、使用方法
使用HttpClient发送请求、接收响应很简单,一般需要如下几步即可。
-
创建HttpClient对象。
-
创建请求方法的实例,并指定请求URL。如果需要发送GET请求,创建HttpGet对象;如果需要发送POST请求,创建HttpPost对象。
-
如果需要发送请求参数,可调用HttpGet、HttpPost共同的setParams(HetpParams params)方法来添加请求参数;对于HttpPost对象而言,也可调用setEntity(HttpEntity entity)方法来设置请求参数。
-
调用HttpClient对象的execute(HttpUriRequest request)发送请求,该方法返回一个HttpResponse。
-
调用HttpResponse的getAllHeaders()、getHeaders(String name)等方法可获取服务器的响应头;调用HttpResponse的getEntity()方法可获取HttpEntity对象,该对象包装了服务器的响应内容。程序可通过该对象获取服务器的响应内容。
-
释放连接。无论执行方法是否成功,都必须释放连接
四、实例:调用快递订单接口
1 首先导入httpclientUtil的工具类
package com.jk.test;
import java.net.URI;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Set;
import org.apache.http.HttpEntity;
import org.apache.http.NameValuePair;
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.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
public class HttpClientUtil {
static CloseableHttpClient client = null;
static {
client = HttpClients.createDefault();
}
/**
* httpClient get请求方法
* @param url 请求地址
* @param params 请求参数
* @return
*/
public static String get(String url,HashMap<String, Object> params){
try {
HttpGet httpGet = new HttpGet();
// map集合.keySet() 方法,拿到map集合中的key
// map.get(key)
// params = [name:zhangsan,age:23]
// keySet: name age
Set<String> keySet = params.keySet();
StringBuffer stringBuffer = new StringBuffer();
// url: 127.0.0.1:8080/aaa/bbb.do
// ?t=12345679000&name=zhangsan&age=23
stringBuffer.append(url).append("?t=").append(System.currentTimeMillis());
for (String key : keySet) {
stringBuffer.append("&").append(key).append("=").append(params.get(key));
}
// 127.0.0.1:8080/aaa/bbb.do?t=12345679000&name=zhangsan&age=23
httpGet.setURI(new URI(stringBuffer.toString()));
// 发起请求,返回响应
CloseableHttpResponse execute = client.execute(httpGet);
int statusCode = execute.getStatusLine().getStatusCode();
// 如果状态码等于200 请求成功
if (200 != statusCode) {
return "";
}
//execute.getEntity() 获取响应里面的实体对象
return EntityUtils.toString(execute.getEntity(), "utf-8");
}catch (Exception e) {
e.printStackTrace();
return null;
}
}
/**
* 127.0.0.1:8080/aaa/bbb.do/zhangsan/23
* get请求 参数为/传递
* @param url 请求地址
* @param params String类型数组形式 String... 变量名
* @return
*/
public static String get(String url, String...params){
try {
HttpGet httpGet = new HttpGet();
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append(url);
for (String string : params) {
stringBuffer.append("/").append(string);
}
httpGet.setURI(new URI(stringBuffer.toString()));
CloseableHttpResponse execute = client.execute(httpGet);
int statusCode = execute.getStatusLine().getStatusCode();
if (200 != statusCode) {
return "";
}
return EntityUtils.toString(execute.getEntity(), "utf-8");
}catch (Exception e) {
e.printStackTrace();
return null;
}
}
public static String post(String url,HashMap<String, Object> params) {
try {
HttpPost httpPost = new HttpPost();
httpPost.setURI(new URI(url));
List<NameValuePair> parameters = new ArrayList<NameValuePair>();
Set<String> keySet = params.keySet();
for (String key : keySet) {
NameValuePair e = new BasicNameValuePair(key, params.get(key).toString());
parameters.add(e);
}
HttpEntity entity = new UrlEncodedFormEntity(parameters , "utf-8");
httpPost.setEntity(entity);
CloseableHttpResponse execute = client.execute(httpPost);
int statusCode = execute.getStatusLine().getStatusCode();
if (200 != statusCode) {
return "";
}
return EntityUtils.toString(execute.getEntity(), "utf-8");
}catch (Exception e) {
e.printStackTrace();
return null;
}
}
public static void main(String[] args) {
//请求网页的路径
String url = "http://t.weather.sojson.com/api/weather/city";
String[] paramArr = {"101010100"};
String val = HttpClientUtil.get(url, paramArr);
//存取{}的格式
JSONObject parseObject = JSONObject.parseObject(val);
JSONObject object = parseObject.getJSONObject("data");
//存取[]的格式
JSONArray jsonArray = object.getJSONArray("forecast");
Object object2 = jsonArray.getJSONObject(0).get("high");
System.out.println(object2);
}
}
2 然后创建快递订单的实体类
package com.jk.yjj.model.user;
public class ExpressBean {
private String time;
private String ftime;
private String context;
private String location;
public String getTime() {
return time;
}
public void setTime(String time) {
this.time = time;
}
public String getFtime() {
return ftime;
}
public void setFtime(String ftime) {
this.ftime = ftime;
}
public String getContext() {
return context;
}
public void setContext(String context) {
this.context = context;
}
public String getLocation() {
return location;
}
public void setLocation(String location) {
this.location = location;
}
@Override
public String toString() {
return "ExpressBean [time=" + time + ", ftime=" + ftime + ", context=" + context + ", location=" + location
+ "]";
}
}
3 在redis缓存中加入
/**
* 物流接口网址
/
public static final String EXPRESS_URL = “http://www.kuaidi100.com/query”;
/*
* 物流接口响应成功状态码
*/
public static final String EXPRESS_STATUS = “200”;
4 调用controller层执行方法调用订单查询方法
package com.jk.yjj.controller.robot;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.jk.yjj.CommonConf;
import com.jk.yjj.model.user.ExpressBean;
import com.jk.yjj.service.robot.RobotService;
import com.jk.yjj.utils.HttpClientUtil;
@Controller
@RequestMapping(“robot”)
public class RobotController {
private static final String String = null;
@RequestMapping("express")
@ResponseBody
public List<ExpressBean> express(String orderId,String type){
HashMap<String, Object> params = new HashMap<>();
params.put("type", type);
params.put("postid", orderId);
String string = HttpClientUtil.get(CommonConf.EXPRESS_URL, params);
JSONObject parseObject = JSON.parseObject(string);
String status = parseObject.getString("status");
if (!status.equals(CommonConf.EXPRESS_STATUS)) {
return new ArrayList<>();
}
String data = parseObject.getString("data");
List<ExpressBean> parseArray = parseObject.parseArray(data,ExpressBean.class);
return parseArray;
}
}
5.前台展示jsp展示订单查询列表
<%@ page language=“java” contentType=“text/html; charset=UTF-8”
pageEncoding=“UTF-8”%>
<div data-options="region:'north',title:'高级搜索'" style="height:20%">
<select id="typeSelect" class="easyui-combobox">
<option value="">--请选择--</option>
<option value="shentong">申通</option>
<option value="ems">EMS</option>
<option value="shunfeng">顺丰</option>
<option value="yuantong">圆通</option>
<option value="zhongtong">中通</option>
<option value="yunda">韵达</option>
<option value="tiantian">天天</option>
<option value="huitongkuaidi">汇通</option>
<option value="quanfengkuaidi">全峰</option>
<option value="debangwuliu">德邦</option>
<option value="zhaijisong">宅急送</option>
</select>
<input id="orderId" type="text" class="easyui-textbox"/>
<a id="searchBtn" class="easyui-linkbutton" data-options="iconCls:'icon-search'">查询</a>
</div>
<div data-options="region:'center',title:'物流信息'" style="height:80%">
<div id="content">
</div>
</div>
</div>