java根据ip地址获取用户地理位置

向右看齐 2024-04-17 06:16 136阅读 0赞
功能需求

工作项目需要,根据用户向服务器请求的ip地址获取用户所在的地理位置。

具体代码

话不多说,直接上代码(作为一个负责任的程序员,我把import内容都贴贴出来了):

  1. /**
  2. * FileName: AddressUtils
  3. * Author: Administrator
  4. * Date: 2019-04-18 14:36
  5. * Description: 通过ip地址获取访问地区
  6. * History:
  7. * <author> <time> <version> <desc>
  8. * 作者姓名:wxz 修改时间: 版本号: 描述:
  9. */
  10. import org.springframework.boot.configurationprocessor.json.JSONObject;
  11. import java.io.BufferedReader;
  12. import java.io.DataOutputStream;
  13. import java.io.IOException;
  14. import java.io.InputStreamReader;
  15. import java.io.UnsupportedEncodingException;
  16. import java.net.HttpURLConnection;
  17. import java.net.URL;
  18. /**
  19. * 根据IP地址获取详细的地域信息
  20. * 淘宝API : http://ip.taobao.com/service/getIpInfo.php?ip=218.192.3.42
  21. * 新浪API : http://int.dpool.sina.com.cn/iplookup/iplookup.php?format=json&ip=218.192.3.42
  22. */
  23. public class AddressUtils {
  24. /**
  25. * @param content 请求的参数 格式为:ip=""
  26. * @return
  27. * @throws UnsupportedEncodingException
  28. * @param//encoding 服务器端请求编码。如GBK,UTF-8等
  29. */
  30. public static String getAddresses(String content, String encodingString)
  31. throws UnsupportedEncodingException {
  32. // 这里调用淘宝API
  33. String urlStr = "http://ip.taobao.com/service/getIpInfo.php";
  34. // 从http://whois.pconline.com.cn取得IP所在的省市区信息
  35. String returnStr = getResult(urlStr, content, encodingString);
  36. if (returnStr != null) {
  37. // 处理返回的省市区信息
  38. System.out.println("(1) unicode转换成中文前的returnStr : " + returnStr);
  39. returnStr = decodeUnicode(returnStr);
  40. System.out.println("(2) unicode转换成中文后的returnStr : " + returnStr);
  41. String[] temp = returnStr.split(",");
  42. if (temp.length < 3) {
  43. return "0";//无效IP,局域网测试
  44. }
  45. return returnStr;
  46. }
  47. return null;
  48. }
  49. /**
  50. * @param urlStr 请求的地址
  51. * @param content 请求的参数 格式为:name=xxx&pwd=xxx
  52. * @param encoding 服务器端请求编码。如GBK,UTF-8等
  53. * @return
  54. */
  55. private static String getResult(String urlStr, String content, String encoding) {
  56. URL url = null;
  57. HttpURLConnection connection = null;
  58. try {
  59. url = new URL(urlStr);
  60. connection = (HttpURLConnection) url.openConnection();// 新建连接实例
  61. connection.setConnectTimeout(8000);// 设置连接超时时间,单位毫秒
  62. connection.setReadTimeout(8000);// 设置读取数据超时时间,单位毫秒
  63. connection.setDoOutput(true);// 是否打开输出流 true|false
  64. connection.setDoInput(true);// 是否打开输入流true|false
  65. connection.setRequestMethod("POST");// 提交方法POST|GET
  66. connection.setUseCaches(false);// 是否缓存true|false
  67. connection.connect();// 打开连接端口
  68. DataOutputStream out = new DataOutputStream(connection
  69. .getOutputStream());// 打开输出流往对端服务器写数据
  70. out.writeBytes(content);// 写数据,也就是提交你的表单 name=xxx&pwd=xxx
  71. out.flush();// 刷新
  72. out.close();// 关闭输出流
  73. BufferedReader reader = new BufferedReader(new InputStreamReader(
  74. connection.getInputStream(), encoding));// 往对端写完数据对端服务器返回数据
  75. // ,以BufferedReader流来读取
  76. StringBuffer buffer = new StringBuffer();
  77. String line = "";
  78. while ((line = reader.readLine()) != null) {
  79. buffer.append(line);
  80. }
  81. reader.close();
  82. return buffer.toString();
  83. }
  84. catch (IOException e) {
  85. e.printStackTrace();
  86. }
  87. finally {
  88. if (connection != null) {
  89. connection.disconnect();// 关闭连接
  90. }
  91. }
  92. return null;
  93. }
  94. /**
  95. * unicode 转换成 中文
  96. *
  97. * @param theString
  98. * @return
  99. */
  100. public static String decodeUnicode(String theString) {
  101. char aChar;
  102. int len = theString.length();
  103. StringBuffer outBuffer = new StringBuffer(len);
  104. for (int x = 0; x < len; ) {
  105. aChar = theString.charAt(x++);
  106. if (aChar == '\\') {
  107. aChar = theString.charAt(x++);
  108. if (aChar == 'u') {
  109. int value = 0;
  110. for (int i = 0; i < 4; i++) {
  111. aChar = theString.charAt(x++);
  112. switch (aChar) {
  113. case '0':
  114. case '1':
  115. case '2':
  116. case '3':
  117. case '4':
  118. case '5':
  119. case '6':
  120. case '7':
  121. case '8':
  122. case '9':
  123. value = (value << 4) + aChar - '0';
  124. break;
  125. case 'a':
  126. case 'b':
  127. case 'c':
  128. case 'd':
  129. case 'e':
  130. case 'f':
  131. value = (value << 4) + 10 + aChar - 'a';
  132. break;
  133. case 'A':
  134. case 'B':
  135. case 'C':
  136. case 'D':
  137. case 'E':
  138. case 'F':
  139. value = (value << 4) + 10 + aChar - 'A';
  140. break;
  141. default:
  142. throw new IllegalArgumentException(
  143. "Malformed encoding.");
  144. }
  145. }
  146. outBuffer.append((char) value);
  147. }
  148. else {
  149. if (aChar == 't') {
  150. aChar = '\t';
  151. }
  152. else if (aChar == 'r') {
  153. aChar = '\r';
  154. }
  155. else if (aChar == 'n') {
  156. aChar = '\n';
  157. }
  158. else if (aChar == 'f') {
  159. aChar = '\f';
  160. }
  161. outBuffer.append(aChar);
  162. }
  163. }
  164. else {
  165. outBuffer.append(aChar);
  166. }
  167. }
  168. return outBuffer.toString();
  169. }
如何调用
  1. // 测试
  2. public static void main(String[] args) throws Exception {
  3. AddressUtils addressUtils = new AddressUtils();
  4. String ip = "222.221.10.111";
  5. String address = "";
  6. try {
  7. address = addressUtils.getAddresses("ip=" + ip, "utf-8");
  8. }
  9. catch (UnsupportedEncodingException e) {
  10. e.printStackTrace();
  11. }
  12. JSONObject jsonObject = new JSONObject(address);
  13. jsonObject = jsonObject.getJSONObject("data");
  14. //通过相应的get方法,获取相应的属性
  15. String county = jsonObject.getString("country_id");//国家
  16. String region = jsonObject.getString("region");//省份
  17. String city = jsonObject.getString("city");//城市
  18. String telecom = jsonObject.getString("isp");//运营商
  19. System.out.println("获取的地址为:" + address);
  20. System.out.println("解析得到的地址为:" +county+","+ region + "省;" + city + "市,"+telecom);
  21. }
效果图

在这里插入图片描述

注意事项

案例中使用的解析地址是淘宝的,也提供了一个新浪的,这些都是免费的api。但是,我说但是了啊,这些免费的api使用起来有一个bug;那就是:如果你频繁调用这个接口,会出现502,调用失败的显现。那么,如何解决这个问题呢?好说,充钱。没错,就是充钱购买比如阿里云提供的一个这样的工具,收费的;
当然,除了充钱,也还有第二个解决方案。在说第二种解决方案之前,我们先来了解一下上面的第一种解决方案的思路:

实现思路

将全球申请的ip地址信息收集起来,做成一个数据库系统,存入服务器端。当需要使用查询的时候,直接使用IP地址和数据库系统中的数据进行对比(因为ip地址在全球都是唯一的,在存入数据库之前,会将IP地址转为一个二进制数字然后进行存储,需要查询的时候也同样是将提供IP地址转为一个二进制数字再进行查询)。

第二种解决方案

在大致了解了这个功能的实现过程以后,我们对于第二种解决方案应该也成竹在胸了。没错,那就是我们自己下载ip地址信息,存储在本地,需要使用的时候,直接在本地查询对比即可。
那么,这个IP地址信息去哪里下载呢?以下是下载地址:http://software77.net/geo-ip/
具体的编写流程由于时间等原因,本次就不做介绍了。下回有时间再来唠叨唠叨

发表评论

表情:
评论列表 (有 0 条评论,136人围观)

还没有评论,来说两句吧...

相关阅读