java调用百度的接口获取起-止位置的距离
需求:校验收货地址是否超出配送范围
重要:
做该需求的思路就是通过卖家和卖家具体的地址信息,来获取到二者的经纬度, 此时可以使用百度的 "地理编码服务",即可获取对应的经纬度
第二步,就是通过二者的经纬度,按照百度接口的要求,发送,即可获取到包含二者距离的JSON串, 此时就可以通过解析jsON获取距离, 最后在判断得到的距离,与自己配送的距离进行比较,即可判断是否超出距离
注册一个百度账号,要求是必须实名认证,需要填写一些基本信息,这里需要注意一下.
首先要获取百度地图的一个AK
登录百度地图开放平台:https://lbsyun.baidu.com/
进入控制台,创建应用,获取AK:
创建应用时:
类型:选服务端
IP白名单:0.0.0.0/0
对于此需求用到了两个百度的接口, 接口地址如下:
地理编码服务: Https://lbsyun.baidu.com/index.PHP?title=webapi/guide/WEBservice-geocoding
https://lbsyun.baidu.com/index.php?title=webapi/directionlite-v1
代码编写:
1. 配置基本属性
sky:
baidumap:
shop-address: 北京市西城区广安门内大街167号翔达大厦1层
ak: XXXXXXXXXXXXXXXXXXXXXXXXXX
default-distance: 5000 // 这里在本文中没有使用,
用于发送请求的工具类
说明,因为现在我们需要从服务器中发送请求,此时我们就需要使用HttpClient这个小框架来实现此功能, 下面的工具类是对此框架的一个封装
package com.sky.utils;
import com.alibaba.fastjson.JSONObject;
import lombok.extern.slf4j.Slf4j;
import org.apache.http.NameValuePair;
import org.apache.http.client.config.RequestConfig;
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.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;
import org.springframework.stereotype.Service;
import java.io.IOException;
import java.net.URI;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
@Slf4j
public class HttpClientUtil {
static final int TIMEOUT_MSEC = 5 * 1000;
public static String doGet(String url, Map<String, String> paramMap) {
// 创建Httpclient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
String result = "";
CloseableHttpResponse response = null;
try {
URIBuilder builder = new URIBuilder(url);
if (paramMap != null) {
for (String key : paramMap.keySet()) {
builder.addParameter(key, paramMap.get(key));
}
}
URI uri = builder.build();
log.info("发送的请求====>{}", uri);
//创建GET请求
HttpGet httpGet = new HttpGet(uri);
//发送请求
response = httpClient.execute(httpGet);
//判断响应状态
if (response.getStatusLine().getStatusCode() == 200) {
result = EntityUtils.toString(response.getEntity(), "UTF-8");
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
response.close();
httpClient.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return result;
}
public static String doPost(String url, Map<String, String> paramMap) throws IOException {
// 创建Httpclient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
CloseableHttpResponse response = null;
String resultString = "";
try {
// 创建Http Post请求
HttpPost httpPost = new HttpPost(url);
// 创建参数列表
if (paramMap != null) {
List<NameValuePair> paramList = new ArrayList();
for (Map.Entry<String, String> param : paramMap.entrySet()) {
paramList.add(new BasicNameValuePair(param.geTKEy(), param.getValue()));
}
// 模拟表单
UrlEncodedFormEntity entity = new UrlEncodedFormEntity(paramList);
httpPost.setEntity(entity);
}
httpPost.setConfig(builderRequestConfig());
// 执行http请求
response = httpClient.execute(httpPost);
resultString = EntityUtils.toString(response.getEntity(), "UTF-8");
} catch (Exception e) {
throw e;
} finally {
try {
response.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return resultString;
}
public static String doPost4Json(String url, Map<String, String> paramMap) throws IOException {
// 创建Httpclient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
CloseableHttpResponse response = null;
String resultString = "";
try {
// 创建Http Post请求
HttpPost httpPost = new HttpPost(url);
if (paramMap != null) {
//构造json格式数据
JSONObject jsonObject = new JSONObject();
for (Map.Entry<String, String> param : paramMap.entrySet()) {
jsonObject.put(param.getKey(), param.getValue());
}
StringEntity entity = new StringEntity(jsonObject.toString(), "utf-8");
//设置请求编码
entity.setContentEncoding("utf-8");
//设置数据类型
entity.setContentType("application/json");
httpPost.setEntity(entity);
}
httpPost.setConfig(builderRequestConfig());
// 执行http请求
response = httpClient.execute(httpPost);
resultString = EntityUtils.toString(response.getEntity(), "UTF-8");
} catch (Exception e) {
throw e;
} finally {
try {
response.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return resultString;
}
private static RequestConfig builderRequestConfig() {
return RequestConfig.custom()
.setConnectTimeout(TIMEOUT_MSEC)
.setConnectionRequestTimeout(TIMEOUT_MSEC)
.setSocketTimeout(TIMEOUT_MSEC).build();
}
}
定义一个Location类用来存放地址的经纬度信息
package com.sky.entity;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class Location {
private double lat;
private double lng;
}
自定义一个工具类,封装对百度接口的请求,方便用于以后在Service层中能够直接的调用 .
注: 因为工具列是自己写的可能会有很多不合适的地方如有发现希望指出
另外其中有的异常类也是自定义如果没有,改为RuntimeException 即可
package com.sky.utils;
import com.sky.entity.Location;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.json.JSONException;
import org.json.JSONObject;
import org.springframework.beans.factory.annotation.Value;
import java.util.HashMap;
@AllArgsConstructor
@NoArgsConstructor
@Data
@Slf4j
public class BaiduMapUtil {
// 获取配置类中的值
@Value("${sky.baidumap.shop-address}")
private String myShopAddress;
@Value("${sky.baidumap.ak}")
private String ak;
public Location getLocation(String userAddress) {
String URL = "https://api.map.baidu.com/geocoding/v3";
HashMap<String, String> map = new HashMap<>();
map.put("address", userAddress);
map.put("output", "json");
map.put("ak", ak);
String body = HttpClientUtil.doGet(URL, map);
Location location = new Location();
try {
JSONObject jsonObject = new JSONObject(body);
// 获取Status
String status = jsonObject.getString("status");
if ("0".equals(status)) {
// 解析JSON
JSONObject res = jsonObject.getJSONObject("result").getJSONObject("location");
// 获取经度
String lng = res.getString("lng");
Double transferLnf = Double.parseDouble(lng);
location.setLng(transferLnf);
// 获取纬度
String lat = res.getString("lat");
Double transferLat = Double.parseDouble(lat);
location.setLat(transferLat);
} else {
// 如果没有返回排除异常交给全局异常处理
throw new RuntimeException("无权限");
}
} catch (Exception e) {
log.info("解析JSON异常,异常信息{}", e.getMessage());
}
return location;
}
public String getDistance(Location userLocation) {
Location myShopLocation = getLocation(myShopAddress);
// 起始位置, 即我的位置
String origin = myShopLocation.getLat() + "," + myShopLocation.getLng();
// 最终位置, 即终点
String destination = userLocation.getLat() + "," + userLocation.getLng();
String url = "https://api.map.baidu.com/directionlite/v1/riding";
// 发送Get请求
HashMap<String, String> map = new HashMap<>();
map.put("origin", origin);
map.put("destination", destination);
map.put("ak", ak);
map.put("steps_info", "0");
String result = HttpClientUtil.doGet(url, map);
String distance = null;
try {
JSONObject jsonObject = new JSONObject(result);
distance = jsonObject.getJSONObject("result").getJSONArray("routes").getJSONObject(0).getString("distance");
} catch (JSONException e) {
log.info("路径异常");
}
log.info("二者距离{}", distance);
return distance;
}
}
此时就可以通过调用工具类传入userAddress
用户的地址, 因为商家的地址已经配置,此时就可以通过调用getDistance
方法获取到二者的距离.
到此这篇关于java调用百度的接口获取起-止位置的距离的文章就介绍到这了,更多相关java调用百度接口获取起-止位置内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
相关文章