北斗定位
Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

250 lignes
10KB

  1. package com.telpo.dipperposition.service.impl;
  2. import com.alibaba.fastjson.JSONObject;
  3. import com.telpo.dipperposition.common.*;
  4. import com.telpo.dipperposition.entity.mongo.IPProvinceEntity;
  5. import com.telpo.dipperposition.service.IDipperAstPosAsyncTaskService;
  6. import com.telpo.dipperposition.service.IpProvinceService;
  7. import lombok.extern.slf4j.Slf4j;
  8. import org.apache.commons.lang3.ObjectUtils;
  9. import org.springframework.beans.factory.annotation.Autowired;
  10. import org.springframework.scheduling.annotation.Async;
  11. import org.springframework.stereotype.Service;
  12. import java.io.UnsupportedEncodingException;
  13. import java.util.List;
  14. /**
  15. * @program: DipperAstPosAsyncTaskServiceImpl
  16. * @description: 系统预先基于省份的省会城市的经纬度作为辅助信息,
  17. * * 根据设备请求的IP地址,从高德IP定位服务获取相关的省份,再匹配相应的位置信息作为辅助位置信息。
  18. * * 如果匹配不到省份,则以武汉中心作为辅助为位置信息。
  19. * * 关于IP与省份的关系保存到缓存中,用于下次使用时,先在缓存中获取匹配信息,匹配不到,再请求高德IP定位服务。
  20. * * 高德IP定位服务:https://lbs.amap.com/api/webservice/guide/api/ipconfig。
  21. * @author: king
  22. * @create: 2021-01-10 14:01
  23. */
  24. @Service
  25. @Slf4j
  26. public class DipperAstPosAsyncTaskServiceImpl implements IDipperAstPosAsyncTaskService {
  27. @Autowired
  28. private RedisUtil redisUtil;
  29. @Autowired
  30. private OkHttpUtil okHttpUtil;
  31. @Autowired
  32. private IpProvinceService ipProvinceService;
  33. private static IDipperAstPosAsyncTaskService dipperAstPosAsyncTaskService;
  34. /**
  35. * spring会自动从ioc容器当中根据IDipperAstPosAsyncTaskService类型找到dipperAstPosAsyncTaskService,当做参数传进来
  36. *
  37. */
  38. @Override
  39. public void setDipperAstPosAsyncTaskService(IDipperAstPosAsyncTaskService dipperAstPosAsyncTaskService) {
  40. this.dipperAstPosAsyncTaskService = dipperAstPosAsyncTaskService;
  41. }
  42. private String centerProvince;
  43. private String centerProvinceFilePath;
  44. private String ipPositionRequestPath;
  45. private String ipPositionRequestKey;
  46. // private String getAstPos(String ipAddress) throws UnsupportedEncodingException {
  47. //
  48. // String centerAddress = getIpPositionProvince(ipAddress);
  49. // if (ObjectUtils.isEmpty(centerAddress) || centerAddress.equals("0")) {
  50. // log.warn("IP地址非法,无法获取辅助位置信息!");
  51. // // 返回武汉的定位数据
  52. // centerAddress = centerProvince;
  53. // } else {
  54. // // 保存到mongoDB
  55. // createIPProvince(ipAddress, centerAddress);
  56. // }
  57. //
  58. // String lonAndAlt;
  59. // if (redisUtil.hasKey(centerAddress)) {
  60. // // 获取省会城市定位信息
  61. // lonAndAlt= (String) redisUtil.get(centerAddress);
  62. // } else {
  63. // // 请求高德IP定位服务
  64. // this.getPosFromFile(centerAddress);
  65. // lonAndAlt = (String) redisUtil.get(centerAddress);
  66. // }
  67. //
  68. // return lonAndAlt;
  69. // }
  70. // 从CSV文件读取省会城市中心点位置信息
  71. private void getPosFromFile(String centerAddress) {
  72. // 不存在说明token是已过期了
  73. String centerProvinceName = "";
  74. String centerProvinceLonAndAlt = "";
  75. List<String> centerAddressSets = CSVUtil.readCSV(this.centerProvinceFilePath);
  76. for (String centerAddressSet:centerAddressSets) {
  77. String[] centerAddressSetArray = centerAddressSet.split(",");
  78. if (centerAddressSetArray.length < 3) {
  79. log.warn("CSV数据格式错误");
  80. } else {
  81. centerProvinceName = centerAddressSetArray[3];
  82. centerProvinceLonAndAlt = centerAddressSetArray[1]+","+centerAddressSetArray[2];
  83. redisUtil.set(centerProvinceName, centerProvinceLonAndAlt, 0);
  84. }
  85. }
  86. }
  87. // 根据IP获取省会信息
  88. private String getIpPositionProvince(String ipAddress) {
  89. // 关于IP与省份的关系保存到缓存中
  90. // 使用时,先在缓存中获取匹配信息
  91. // 用mongodb实现
  92. IPProvinceEntity ipProvinceEntity = ipProvinceService.getIpProvince(ipAddress);
  93. if (ipProvinceEntity == null) {
  94. // 匹配不到,再请求高德IP定位服务。
  95. JSONObject userObj = new JSONObject();
  96. userObj.put("ip", ipAddress);
  97. userObj.put("key", ipPositionRequestKey);
  98. JSONObject json = okHttpUtil.postRequestWithJson(ipPositionRequestPath, null, userObj);
  99. if (ObjectUtils.isNotEmpty(json)) {
  100. String province = (String) json.get("province");
  101. if (ObjectUtils.isEmpty(province)) {
  102. log.debug("json is :" + json.toString());
  103. return null;
  104. }
  105. return province;
  106. } else {
  107. // 意外错误
  108. log.debug("ip address is null");
  109. return null;
  110. }
  111. } else {
  112. return ipProvinceEntity.getProvince();
  113. }
  114. }
  115. // 将IP对应的省会保存到mongoDB
  116. @Async("asyncServiceExecutor")
  117. public void createIPProvince(String ipAddress, String province) {
  118. log.debug("异步创建推送失败任务记录!");
  119. try {
  120. IPProvinceEntity ipProvinceEntity = ipProvinceService.getIpProvince(ipAddress);
  121. if (ipProvinceEntity == null) {
  122. //DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
  123. ipProvinceEntity.setIp(ipAddress);
  124. ipProvinceEntity.setProvince(province);
  125. ipProvinceService.saveIpProvince(ipProvinceEntity);
  126. // } else {
  127. //
  128. // ipProvinceEntity.setProvince(province);
  129. // iPProvinceService.updateIPProvince(ipProvinceEntity, ipProvinceEntity);
  130. }
  131. } catch (Exception e) {
  132. log.error("创建推送失败记录异常:", e);
  133. }
  134. }
  135. /*
  136. * 获取定位辅助信息
  137. * @param ipAddress
  138. */
  139. @Override
  140. @Async("asyncServiceExecutor")
  141. public String pushAstPos(String ipAddress,
  142. String centerProvinceFilePath,
  143. String centerProvince,
  144. String ipPositionRequestPath,
  145. String ipPositionRequestKey) throws UnsupportedEncodingException {
  146. this.ipPositionRequestKey = ipPositionRequestKey;
  147. this.ipPositionRequestPath = ipPositionRequestPath;
  148. this.centerProvince = centerProvince;
  149. this.centerProvinceFilePath = centerProvinceFilePath;
  150. // (1) 获取省会城市信息
  151. String centerAddress = getIpPositionProvince(ipAddress);
  152. if (ObjectUtils.isEmpty(centerAddress) || centerAddress.equals("0")) {
  153. log.warn("IP地址非法,无法获取辅助位置信息!");
  154. // 返回武汉的定位数据
  155. centerAddress = this.centerProvince;
  156. } else {
  157. // 保存到mongoDB
  158. createIPProvince(ipAddress, centerAddress);
  159. }
  160. String lonAndAlt;
  161. if (redisUtil.hasKey(centerAddress)) {
  162. // 获取省会城市定位信息
  163. lonAndAlt= (String) redisUtil.get(centerAddress);
  164. } else {
  165. // 请求高德IP定位服务
  166. this.getPosFromFile(centerAddress);
  167. lonAndAlt = (String) redisUtil.get(centerAddress);
  168. }
  169. // (2) 处理返回结果
  170. if (lonAndAlt == null) {
  171. // null处理
  172. log.error("系统错误,请联系系统管理员。");
  173. return null;
  174. //return;
  175. }
  176. // push to GNNS Server
  177. String pushResult = getCmdOfPos(lonAndAlt);
  178. return pushResult;
  179. }
  180. // 组装命令发送给设备
  181. private String getCmdOfPos(String astPos) {
  182. // 创建Socket客户端实例;
  183. // SocketClient client = new SocketClient(astServer, posAstPort, astTimeout);
  184. // 时间和位置不是从服务器获取,而是本地生成
  185. String[] astPosArray = astPos.split(",");
  186. String lan = astPosArray[0].trim();
  187. String alt = astPosArray[1].trim();
  188. double lanValue = Double.parseDouble(lan) * 10000000;
  189. long lanLongValue = Double.doubleToLongBits(lanValue);
  190. if (lanLongValue < 0) {
  191. lanLongValue = lanLongValue + 4294967295L + 1;
  192. }
  193. double altValue = Double.parseDouble(alt) * 10000000;
  194. long altLongValue = Double.doubleToLongBits(altValue);
  195. if (altLongValue < 0) {
  196. altLongValue = altLongValue + 4294967295L + 1;
  197. }
  198. // 数值换算举例(以经度举例。纬度、高度、位置精度换算方法一致):
  199. // (1)经度数值为 113.431,则换算方法如下:
  200. // 113.431/比例因子 = 1134310000(十进制)
  201. //  439C3270(十六进制)
  202. //  经度数据填入 70 32 9C 43(小端模式)
  203. // (2)经度数值为-113.431,则换算方法如下:
  204. // 113.431/比例因子 = 1134310000(十进制)
  205. //  439C3270(十六进制)
  206. //  FFFFFFFF - 439C3270 + 1= BC63CD90(补码)
  207. // 经度数据填入 90 CD 63 BC(小端模式)
  208. // 指令(十六进制)
  209. // 举例: 23 3E 04 01 10 00 70 32 9C 43 D0 B2 CE 0D 70 17 00 00 40 0D 03 00 CA 95
  210. // 其中
  211. // 23 3E 为同步头
  212. // 04 01 为识别码
  213. // 10 00 表示长度为 16
  214. // 70 32 9C 43 表示注入的辅助经度为 113.431 度
  215. // D0 B2 CE 0D 表示注入的辅助纬度为 23.165 度
  216. // 00 2F 为校验和
  217. // astTimeCmd 组装
  218. String astTimeCmd = "233E0401";
  219. astTimeCmd += "1000";
  220. astTimeCmd += HexConvert.encodeHEX(lanLongValue);
  221. astTimeCmd += HexConvert.encodeHEX(altLongValue);
  222. //String hexIn = HexConvert.convertStringToHex(astTimeCmd) + HexConvert.makeChecksum(astTimeCmd);
  223. String hexIn = astTimeCmd + HexConvert.makeChecksum(astTimeCmd);
  224. //return sendResult;
  225. return hexIn;
  226. }
  227. }