前几个月微信公众号上线了IP归属地
的功能,后续知乎、抖音等平台纷纷添加了该功能。如果是国内的用户精确到省份,国外用户精确到国家。本文就使用Java
实现获取IP归属地
。
主要讲解几个步骤:
Java
获取请求IP
解决Nginx
转发问题
通过IP
地址获取归属地
获取 IP 地址
首先使用基于Spring Boot
搭建项目,在controller
添加HttpServletRequest
请求参数:
@RestController
public class IpController {
@GetMapping("/ip-address")
public String ipAddress(HttpServletRequest request) {
// 接收request
}
}
复制代码
通过HttpServletRequest
获取IP地址
:
String ip = request.getHeader("x-forwarded-for");
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("Proxy-Client-IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("WL-Proxy-Client-IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("HTTP_CLIENT_IP");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getHeader("HTTP_X_FORWARDED_FOR");
}
if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
ip = request.getRemoteAddr();
}
return ip;
复制代码
在本地环境调用获取 IP,要么是0:0:0:0:0:0:0:1
,或者是局域网IP
。
局域网IP
是以192.168.x.x
开头,或者是127.0.0.1
的IP
。
所以需要部署到外网服务器才能获取到公网地址。部署到外网服务器能成功获取IP
地址。
Nginx 反向代理问题
直接访问公网服务器地址能成功获取IP
地址,但是通过Nginx
反向代理获取的都是127.0.0.1
。客户端请求Nginx
服务器再反向代理转发到服务端,此时拿到的IP
反向代理的IP
,也就是Nginx
服务器的IP
,并不是真正的客户端IP
。
在Nginx
的配置文件中的location
模块添加以下配置,将客户端的IP
传入到Nginx
服务:
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
复制代码
示例:
server {
listen 80;
server_name localhost;
location / {
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_pass http://xxxx;
}
复制代码
完成以上操作之后,就能成功获取到IP
了。然后通过IP
获取归属地了。
IP 获取归属地
通过IP
获取归属地一般都从地址库找到匹配的地址,本文介绍两种方法.
通过归属地 API 获取
需要发起http
请求,这里使用Spring Boot
的RestTemplate
发起http
请求,首先创建RestTemplate
的bean
实例:
@Configuration
public class RestTemplateConfig {
@Bean
public RestTemplate restTemplate() {
return new RestTemplate();
}
}
复制代码
再调用RestTemplate
发起http
请求:
private String URL = "https://api.beijinxuetang.com/api/common/ip";
JSONObject jsonObject = new JSONObject();
jsonObject.put("ip",ip);
JSONObject json = restTemplate.postForObject(URL,jsonObject, JSONObject.class);
if (json.getInteger("code") == 0) {
json = json.getJSONObject("data");
// 国家
String nation = json.getString("nation");
// 省份
String province = json.getString("province");
// 市
String city = json.getString("city");
}
复制代码
上面的json
是引入fastjson
。
通过地址库获取
使用API接口
,可能会出现服务挂了,或者服务地址不提供服务了等问题。而采用本地地址库就没有这些问题。
本文采用离线IP
地址定位库 Ip2region,Ip2region
是一个离线IP
地址定位库,微秒的查询时间:
首先找到在gihub官网找到地址库ip2region.xdb
,具体路径为data/ip2region.xdb
:
将ip2region.xdb
放在项目的resources
目录下:
引入maven
依赖:
<dependency>
<groupId>org.lionsoul</groupId>
<artifactId>ip2region</artifactId>
<version>2.6.5</version>
</dependency>
复制代码
获取归属地:
private Searcher searcher;
@Override
public String getIpAddress(String ip){
if ("127.0.0.1".equals(ip) || ip.startsWith("192.168")) {
return "局域网 ip";
}
if (searcher == null) {
try {
File file = ResourceUtils.getFile("classpath:ipdb/ip2region.xdb");
String dbPath = file.getPath();
searcher = Searcher.newWithFileOnly(dbPath);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
String region = null;
String errorMessage = null;
try {
region = searcher.search(ip);
} catch (Exception e) {
errorMessage = e.getMessage();
if (errorMessage != null && errorMessage.length() > 256) {
errorMessage = errorMessage.substring(0,256);
}
e.printStackTrace();
}
// 输出 region
}
复制代码
获取region
就能获取到IP
归属地了。例如中国|0|广东省|广州市|电信
。
小程序效果展示
根据上面的程序,做了一个小程序展示归属地。
页面效果图:
扫一扫,就能获取查到自己的归属地了。
评论