php中文网 | cnphp.com

 找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 417|回复: 0

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

[复制链接]

3142

主题

3152

帖子

1万

积分

管理员

Rank: 9Rank: 9Rank: 9

UID
1
威望
0
积分
7956
贡献
0
注册时间
2021-4-14
最后登录
2024-11-22
在线时间
763 小时
QQ
发表于 2022-9-12 07:16:31 | 显示全部楼层 |阅读模式
前几个月微信公众号上线了IP归属地的功能,后续知乎、抖音等平台纷纷添加了该功能。如果是国内的用户精确到省份,国外用户精确到国家。本文就使用Java实现获取IP归属地。
da3cd64154dd3a2641aaade32b7222ba_2448954-20220911232738542-1224303725.png

主要讲解几个步骤:

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地址定位库,微秒的查询时间:
5e4a74aab810290a2019a7717c6d451c_2448954-20220911232812520-1495171373.png

首先找到在gihub官网找到地址库ip2region.xdb,具体路径为data/ip2region.xdb:
6b024230a3b479d8a786f0f62ca40dd0_2448954-20220911232834994-1190248117.png

将ip2region.xdb放在项目的resources目录下:


22c0a533f7d470e42ea0cbab09422f65_2448954-20220911232900418-574597130.png
引入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|广东省|广州市|电信。

小程序效果展示
根据上面的程序,做了一个小程序展示归属地。
eb20c94ae997ea93ced8d9778b135728_2448954-20220911232924314-278921587.png
页面效果图:

fcfdba3f354d20820ac0fd04f7f6df07_2448954-20220911232941597-686469076.png
扫一扫,就能获取查到自己的归属地了。







上一篇:EntityFrameworkCore 模型自动更新(上)
下一篇:liunx标准输入与输出
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|php中文网 | cnphp.com ( 赣ICP备2021002321号-2 )

GMT+8, 2024-11-22 19:59 , Processed in 0.276094 second(s), 44 queries , Gzip On.

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2020, Tencent Cloud.

申明:本站所有资源皆搜集自网络,相关版权归版权持有人所有,如有侵权,请电邮(fiorkn@foxmail.com)告之,本站会尽快删除。

快速回复 返回顶部 返回列表