手把手教你用Java获取IP归属地

发布于:2022-12-20 ⋅ 阅读:(356) ⋅ 点赞:(0)

前几个月微信公众号上线了 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|广东省|广州市|电信 。

小程序效果展示

根据上面的程序,做了一个小程序展示归属地。

页面效果图:

扫一扫,就能获取查到自己的归属地了。


网站公告

今日签到

点亮在社区的每一天
去签到