如何优雅的调用第三方接口

上一篇:重磅!阿里发布全新操作系统,成功顶住多年双 11,这次要干翻 CentOS 了!

来源:blog.csdn.net/qq_35860138/article/details/82967727

在项目开发中经常会遇到调用第三方接口的情况,比如说调用第三方的天气预报接口。

使用流程

1、准备工作:在项目的工具包下导入HttpClientUtil这个工具类,或者也可以使用Spring框架的restTemplate来调用,上面有调用接口的方法【分为Get和Post方式的有参和无参调用】:

package com.njsc.credit.util;


import java.io.IOException;
import java.net.URI;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;


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.client.utils.URIBuilder;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.StringEntity;
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;


public class HttpClientUtil {


/**
 * 带参数的get请求
 * @param url
 * @param param
 * @return String
 */
public static String doGet(String url, Map<String, String> param) {
// 创建Httpclient对象
CloseableHttpClient httpclient = HttpClients.createDefault();


String resultString = "";
CloseableHttpResponse response = null;
try {
// 创建uri
URIBuilder builder = new URIBuilder(url);
if (param != null) {
for (String key : param.keySet()) {
builder.addParameter(key, param.get(key));
}
}
URI uri = builder.build();
// 创建http GET请求
HttpGet httpGet = new HttpGet(uri);
// 执行请求
response = httpclient.execute(httpGet);
// 判断返回状态是否为200
if (response.getStatusLine().getStatusCode() == 200) {
resultString = EntityUtils.toString(response.getEntity(), "UTF-8");
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (response != null) {
response.close();
}
httpclient.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return resultString;
}


/**
 * 不带参数的get请求
 * @param url
 * @return String
 */
public static String doGet(String url) {
return doGet(url, null);
}


/**
 * 带参数的post请求
 * @param url
 * @param param
 * @return String
 */
public static String doPost(String url, Map<String, String> param) {
// 创建Httpclient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
CloseableHttpResponse response = null;
String resultString = "";
try {
// 创建Http Post请求
HttpPost httpPost = new HttpPost(url);
// 创建参数列表
if (param != null) {
List<NameValuePair> paramList = new ArrayList<>();
for (String key : param.keySet()) {
paramList.add(new BasicNameValuePair(key, param.get(key)));
}
// 模拟表单
UrlEncodedFormEntity entity = new UrlEncodedFormEntity(paramList);
httpPost.setEntity(entity);
}
// 执行http请求
response = httpClient.execute(httpPost);
resultString = EntityUtils.toString(response.getEntity(), "utf-8");
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
response.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return resultString;
}


/**
 * 不带参数的post请求
 * @param url
 * @return String
 */
public static String doPost(String url) {
return doPost(url, null);
}


/**
 * 传送json类型的post请求
 * @param url
 * @param json
 * @return String
 */
public static String doPostJson(String url, String json) {
// 创建Httpclient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
CloseableHttpResponse response = null;
String resultString = "";
try {
// 创建Http Post请求
HttpPost httpPost = new HttpPost(url);
// 创建请求内容
StringEntity entity = new StringEntity(json, ContentType.APPLICATION_JSON);
httpPost.setEntity(entity);
// 执行http请求
response = httpClient.execute(httpPost);
resultString = EntityUtils.toString(response.getEntity(), "utf-8");
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
response.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return resultString;
}
}

2、创建url和访问key 以及参数等:

a8883f514f77dd03b5801efbbd0d4575.png

代码如下:

/**
 * 聚合接口校验身份证
 * @param idCard
 * @param realName
 * @return boolean
 */
public boolean identityCheck(String idCard, String realName){
logger.info("-----------------调用聚合数据 身份证验证API BEGIN--------------->");
String key = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX";
String url = "http://op.juhe.cn/idcard/query" + "?key=" + key + "&idcard=" + idCard + "&realname=" + realName;
logger.info("请求url:" + url);
boolean match = false; //是否匹配
try {
String result = HttpClientUtil.doGet(url);
System.out.println("请求结果:" + result);
IdentityCheckResult identityCheckResult = JsonUtils.parse(result, IdentityCheckResult.class);
IdentityCheck identityCheck = JsonUtils.parse(result, "result", IdentityCheck.class);
logger.info(identityCheckResult);
logger.info(identityCheck.toString());
if(identityCheckResult.correct() && identityCheck.getRes() == 1){
match = true;
}
} catch (Exception e) {
e.printStackTrace();
}
logger.info("<-----------------调用聚合数据 身份证验证API END---------------");
return match;
}

3、请求这个第三方接口:

使用HttpClientUtil工具类中的doGet方法来请求URL,得到结果,现在大多数是一个json字符串,类型为String,搜索公众号互联网架构师复“2T”,送你一份惊喜礼包。

4、根据接口返回数据格式来解析数据:

2588f318923e8ea6c34ed7a2ccc8cb24.png

可以看到,返回参数有六个,所以在项目中新建一个bean,包含以上六个字段,用来接住返回数据,如下:

91a5100aaeeff86cf1b68a775798a60f.png

因为接口返回的数据是一个json的字符串,类型实际上是一个String字符串,要解析数据,用工具类JsonUtils的parse方法将字符串转换为Java对象JsonUtils,的代码如下:

package com.eqianxian.commons.utils.json;


import java.util.List;
import java.util.Map;


import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.serializer.PropertyFilter;
import com.alibaba.fastjson.serializer.SerializerFeature;


/**
 * 在系统中统一使用这个,以方便将来切换不同的JSON生成工具
 * 
 * @author KelvinZ
 * 
 */
public class JsonUtils {
public static final int TYPE_FASTJSON = 0;
public static final int TYPE_GSON = 1;


/**
 * <pre>
 * 对象转化为json字符串
 * 
 * @param obj 待转化对象
 * @return 代表该对象的Json字符串
 */
public static final String toJson(final Object obj) {
return JSON.toJSONString(obj);
// return gson.toJson(obj);
}


/**
 * <pre>
 * 对象转化为json字符串
 * 
 * @param obj 待转化对象
 * @return 代表该对象的Json字符串
 */
public static final String toJson(final Object obj, SerializerFeature... features) {
return JSON.toJSONString(obj, features);
// return gson.toJson(obj);
}


/**
 * 对象转化为json字符串并格式化
 * 
 * @param obj
 * @param format 是否要格式化
 * @return
 */
public static final String toJson(final Object obj, final boolean format) {
return JSON.toJSONString(obj, format);
}


/**
 * 对象对指定字段进行过滤处理,生成json字符串
 * 
 * @param obj
 * @param fields 过滤处理字段
 * @param ignore true做忽略处理,false做包含处理
 * @param features json特征,为null忽略
 * @return
 */
public static final String toJson(final Object obj, final String[] fields, final boolean ignore,
SerializerFeature... features) {
if (fields == null || fields.length < 1) {
return toJson(obj);
}
if (features == null)
features = new SerializerFeature[] { SerializerFeature.QuoteFieldNames };
return JSON.toJSONString(obj, new PropertyFilter() {
@Override
public boolean apply(Object object, String name, Object value) {
for (int i = 0; i < fields.length; i++) {
if (name.equals(fields[i])) {
return !ignore;
}
}
return ignore;
}
}, features);
}


/**
 * <pre>
 * 解析json字符串中某路径的值
 * 
 * @param json
 * @param path
 * @return
 */
@SuppressWarnings("unchecked")
public static final <E> E parse(final String json, final String path) {
String[] keys = path.split(",");
JSONObject obj = JSON.parseObject(json);
for (int i = 0; i < keys.length - 1; i++) {
obj = obj.getJSONObject(keys[i]);
}
return (E) obj.get(keys[keys.length - 1]);
}


/**
 * <pre>
 * json字符串解析为对象
 * 
 * @param json 代表一个对象的Json字符串
 * @param clazz 指定目标对象的类型,即返回对象的类型
 * @return 从json字符串解析出来的对象
 */
public static final <T> T parse(final String json, final Class<T> clazz) {
return JSON.parseObject(json, clazz);
}


/**
 * <pre>
 * json字符串解析为对象
 * 
 * @param json json字符串
 * @param path 逗号分隔的json层次结构
 * @param clazz 目标类
 */
public static final <T> T parse(final String json, final String path, final Class<T> clazz) {
String[] keys = path.split(",");
JSONObject obj = JSON.parseObject(json);
for (int i = 0; i < keys.length - 1; i++) {
obj = obj.getJSONObject(keys[i]);
}
String inner = obj.getString(keys[keys.length - 1]);
return parse(inner, clazz);
}


/**
 * 将制定的对象经过字段过滤处理后,解析成为json集合
 * 
 * @param obj
 * @param fields
 * @param ignore
 * @param clazz
 * @param features
 * @return
 */
public static final <T> List<T> parseArray(final Object obj, final String[] fields, boolean ignore,
final Class<T> clazz, final SerializerFeature... features) {
String json = toJson(obj, fields, ignore, features);
return parseArray(json, clazz);
}


/**
 * <pre>
 * 从json字符串中解析出一个对象的集合,被解析字符串要求是合法的集合类型
 * (形如:["k1":"v1","k2":"v2",..."kn":"vn"])
 * 
 * @param json - [key-value-pair...]
 * @param clazz
 * @return
 */
public static final <T> List<T> parseArray(final String json, final Class<T> clazz) {
return JSON.parseArray(json, clazz);
}


/**
 * <pre>
 * 从json字符串中按照路径寻找,并解析出一个对象的集合,例如:
 * 类Person有一个属性name,要从以下json中解析出其集合:
 * {
 * "page_info":{
 * "items":{
 * "item":[{"name":"KelvinZ"},{"name":"Jobs"},...{"name":"Gates"}]
 * }
 * }
 * 使用方法:parseArray(json, "page_info,items,item", Person.class),
 * 将根据指定路径,正确的解析出所需集合,排除外层干扰
 * 
 * @param json json字符串
 * @param path 逗号分隔的json层次结构
 * @param clazz 目标类
 * @return
 */
public static final <T> List<T> parseArray(final String json, final String path, final Class<T> clazz) {
String[] keys = path.split(",");
JSONObject obj = JSON.parseObject(json);
for (int i = 0; i < keys.length - 1; i++) {
obj = obj.getJSONObject(keys[i]);
}
String inner = obj.getString(keys[keys.length - 1]);
List<T> ret = parseArray(inner, clazz);
return ret;
}


/**
 * <pre>
 * 有些json的常见格式错误这里可以处理,以便给后续的方法处理
 * 常见错误:使用了\" 或者 "{ 或者 }",腾讯的页面中常见这种格式
 * 
 * @param invalidJson 包含非法格式的json字符串
 * @return
 */
public static final String correctJson(final String invalidJson) {
String content = invalidJson.replace("\\\"", "\"").replace("\"{", "{").replace("}\"", "}");
return content;
}


/**
 * 格式化Json
 * 
 * @param json
 * @return
 */
public static final String formatJson(String json) {
Map<?, ?> map = (Map<?, ?>) JSON.parse(json);
return JSON.toJSONString(map, true);
}


/**
 * 获取json串中的子json
 * 
 * @param json
 * @param path
 * @return
 */
public static final String getSubJson(String json, String path) {
String[] keys = path.split(",");
JSONObject obj = JSON.parseObject(json);
for (int i = 0; i < keys.length - 1; i++) {
obj = obj.getJSONObject(keys[i]);
System.out.println(obj.toJSONString());
}
return obj != null ? obj.getString(keys[keys.length - 1]) : null;
}


}

感谢您的阅读,也欢迎您发表关于这篇文章的任何建议,关注我,技术不迷茫!小编到你上高速。

    · END ·

最后,关注公众号互联网架构师,在后台回复:2T,可以获取我整理的 Java 系列面试题和答案,非常齐全。

正文结束

推荐阅读 ↓↓↓

1.不认命,从10年流水线工人,到谷歌上班的程序媛,一位湖南妹子的励志故事

2.如何才能成为优秀的架构师?

3.从零开始搭建创业公司后台技术栈

4.程序员一般可以从什么平台接私活?

5.37岁程序员被裁,120天没找到工作,无奈去小公司,结果懵了...

6.IntelliJ IDEA 2019.3 首个最新访问版本发布,新特性抢先看

7.这封“领导痛批95后下属”的邮件,句句扎心!

8.15张图看懂瞎忙和高效的区别!

8d626f026538b1b6f093a6125dcf80b5.gif

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我来给你一个例子,这个例子演示了如何使用动态代理实现一个远程过程调用(RPC)框架。我们将使用Java的动态代理功能来代理第三方接口调用,实现远程过程调用。 首先,我们需要定义一个接口,这个接口是我们要远程调用的第三方接口,假设这个接口是一个计算器接口,可以进行加、减、乘、除等运算: ```java public interface Calculator { public int add(int a, int b); public int subtract(int a, int b); public int multiply(int a, int b); public int divide(int a, int b) throws Exception; } ``` 然后,我们需要定义一个远程服务,这个服务可以接受远程调用,并将调用结果返回给客户端。假设我们的远程服务是一个简单的HTTP服务,可以接受POST请求,并将请求参数和调用结果以JSON格式返回: ```java public class RemoteService { public String invoke(String url, String method, Object[] args) throws Exception { // 构造请求参数 JSONObject request = new JSONObject(); request.put("method", method); request.put("args", args); // 发送POST请求 HttpClient client = new DefaultHttpClient(); HttpPost post = new HttpPost(url); post.setEntity(new StringEntity(request.toString())); HttpResponse response = client.execute(post); // 解析响应结果 String result = EntityUtils.toString(response.getEntity()); JSONObject json = new JSONObject(result); if (json.getBoolean("success")) { return json.getString("result"); } else { throw new Exception(json.getString("error")); } } } ``` 接下来,我们需要实现一个动态代理类,这个代理类可以代理任意一个实现了Calculator接口的类,将接口方法调用转换为远程调用,并将调用结果返回给客户端。我们可以使用Java的动态代理功能来实现这个代理类: ```java public class CalculatorProxy implements InvocationHandler { private String url; private RemoteService remoteService; public CalculatorProxy(String url) { this.url = url; this.remoteService = new RemoteService(); } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 将方法调用转换为远程调用 String methodName = method.getName(); String result = remoteService.invoke(url, methodName, args); // 解析远程调用结果 JSONObject json = new JSONObject(result); if (json.getBoolean("success")) { return json.getInt("result"); } else { throw new Exception(json.getString("error")); } } public static Calculator createProxy(String url) { return (Calculator) Proxy.newProxyInstance(Calculator.class.getClassLoader(), new Class[] { Calculator.class }, new CalculatorProxy(url)); } } ``` 在上面的代理类中,我们实现了InvocationHandler接口的invoke方法,这个方法会在代理对象上调用任意一个接口方法时被调用。在这个方法中,我们将接口方法调用转换为远程调用,并将调用结果解析为Java对象。最后,我们使用Proxy.newProxyInstance方法创建一个代理对象,并返回给客户端使用。 最后,我们可以在客户端代码中使用这个代理对象来调用远程服务,就像调用本地方法一样: ```java Calculator calculator = CalculatorProxy.createProxy("http://remote-service.com/calculator"); int result = calculator.add(1, 2); ``` 这个例子中,我们演示了如何使用Java的动态代理功能实现一个简单的RPC框架。这个框架可以方便地代理任意一个实现了特定接口的类,并将接口方法调用转换为远程调用。这个框架的代码比较简单,易于理解和扩展,可以作为学习动态代理和RPC的一个好例子。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值