Java 中是如何获取 IP 属地的

 细心的小伙伴可能会发现,抖音新上线了IP属地的功能,小伙伴在发表动态、发表评论以及聊天的时候,都会显示自己的IP属地信息


下面,我就来讲讲,Java中是如何获取IP属地的,主要分为以下几步

通过 HttpServletRequest 对象,获取用户的IP地址

通过 IP 地址,获取对应的省份、城市

首先需要写一个IP获取的工具类,因为每一次用户的Request请求,都会携带上请求的IP地址放到请求头中。

public class IpUtil {

    public static String getIpAddr(ServerHttpRequest request) {

        HttpHeaders headers = request.getHeaders();

        String ipAddress = headers.getFirst("X-Forwarded-For");

        if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {

            ipAddress = headers.getFirst("Proxy-Client-IP");

        }

        if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {

            ipAddress = headers.getFirst("WL-Proxy-Client-IP");

        }

        if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {

            ipAddress = request.getRemoteAddress().getAddress().getHostAddress();

            if (ipAddress.equals("127.0.0.1") || ipAddress.equals("0:0:0:0:0:0:0:1")) {

                // 根据网卡取本机配置的IP

                try {

                    InetAddress inet = InetAddress.getLocalHost();

                    ipAddress = inet.getHostAddress();

                } catch (UnknownHostException e) {

                    log.error("根据网卡获取本机配置的IP异常", e);

                }

            }

        }

        // 对于通过多个代理的情况,第一个IP为客户端真实IP,多个IP按照','分割

        if (ipAddress != null && ipAddress.indexOf(",") > 0) {

            ipAddress = ipAddress.split(",")[0];

        }

        return ipAddress;

    }

}

这里有三个名词,分别是

X-Forwarded-For一个 HTTP扩展头部,主要是为了让Web服务器获取访问用户的真实IP地址。每个IP地址,每个值通过逗号+空格分开,最左边是最原始客户端的IP地址,中间如果有多层代理,每⼀层代理会将连接它的客户端IP追加在X-Forwarded-For右边。

X-Real-IP:一般只记录真实发出请求的客户端IP

Proxy-Client-IP:这个一般是经过Apache http服务器的请求才会有,用Apache http做代理时一般会加上Proxy-Client-IP请求头

WL-Proxy-Client-IP:也是通过 Apache http 服务器,在weblogic插件加上的头。

在我们获取到用户的IP地址后,那么就可以获取对应的ip信息了

我在Github冲浪的时候,发现了Ip2region项目。

一个准确率99.9%的离线IP地址定位库,0.0x毫秒级查询,ip2region.db数据库只有数MB,提供了 java,php,c,python,nodejs,golang,c# 等查询绑定和BinaryB树,内存三种查询算法。

数据聚合了一些知名ip到地名查询提供商的数据,这些是他们官方的的准确率,经测试着实比经典的纯真IP定位准确一些。ip2region的数据聚合自以下服务商的开放API或者数据。

80%, 淘宝IP地址库, http://ip.taobao.com/

≈10%, GeoIP, https://geoip.com/

≈2%, 纯真IP库, http://www.cz88.net/

备注:如果上述开放API或者数据都不给开放数据时ip2region将停止数据的更新服务。

每条ip数据段都固定了格式:

_城市Id|国家|区域|省份|城市|ISP_

只有中国的数据精确到了城市,其他国家有部分数据只能定位到国家,后前的选项全部是0,已经包含了全部你能查到的大大小小的国家

生成的数据库文件ip2region.db只有几MB,最小的版本只有1.5MB,随着数据的详细度增加数据库的大小也慢慢增大,目前还没超过8MB

内置的三种查询算法

全部的查询客户端单次查询都在0.x毫秒级别,内置了三种查询算法

memory算法:整个数据库全部载入内存,单次查询都在0.1x毫秒内,C语言的客户端单次查询在0.00x毫秒级别。

binary算法:基于二分查找,基于ip2region.db文件,不需要载入内存,单次查询在0.x毫秒级别。

b-tree算法:基于btree算法,基于ip2region.db文件,不需要载入内存,单词查询在0.x毫秒级别,比binary算法更快。

ip2region安装

下面,就让我们给项目引入ip2region,进行ip信息转换吧

首先引入maven依赖

<dependency>

    <groupId>org.lionsoul</groupId>

    <artifactId>ip2region</artifactId>

    <version>1.7.2</version>

</dependency>

然后编写一个工具类IpUtils,首先需要加载ip2region.db文件

static {

    dbPath = createFtlFileByFtlArray() + "ip2region.db";

    try {

        config = new DbConfig();

    } catch (DbMakerConfigException e) {

        e.printStackTrace();

    }

    try {

        searcher = new DbSearcher(config, dbPath);

    } catch (FileNotFoundException e) {

        e.printStackTrace();

    }

}

在加载的时候,需要下载仓库中的ip2region.db文件,然后放到resource目录下

然后,通过内置的三种算法,分别转换用户ip地址

    public static String getCityInfo(String ip) {

        if (StringUtils.isEmpty(dbPath)) {

            log.error("Error: Invalid ip2region.db file");

            return null;

        }

        if(config == null || searcher == null){

            log.error("Error: DbSearcher or DbConfig is null");

            return null;

        }

        //查询算法

        //B-tree, B树搜索(更快)

        int algorithm = DbSearcher.BTREE_ALGORITHM;

        //Binary,使用二分搜索

        //DbSearcher.BINARY_ALGORITHM

        //Memory,加载内存(最快)

        //DbSearcher.MEMORY_ALGORITYM

        try {

            // 使用静态代码块,减少文件读取操作

//            DbConfig config = new DbConfig();

//            DbSearcher searcher = new DbSearcher(config, dbPath);

            //define the method

            Method method = null;

            switch (algorithm) {

                case DbSearcher.BTREE_ALGORITHM:

                    method = searcher.getClass().getMethod("btreeSearch", String.class);

                    break;

                case DbSearcher.BINARY_ALGORITHM:

                    method = searcher.getClass().getMethod("binarySearch", String.class);

                    break;

                case DbSearcher.MEMORY_ALGORITYM:

                    method = searcher.getClass().getMethod("memorySearch", String.class);

                    break;

                default:

            }

            DataBlock dataBlock = null;

            if (Util.isIpAddress(ip) == false) {

                System.out.println("Error: Invalid ip address");

            }

            dataBlock = (DataBlock) method.invoke(searcher, ip);

            String ipInfo = dataBlock.getRegion();

            if (!StringUtils.isEmpty(ipInfo)) {

                ipInfo = ipInfo.replace("|0", "");

                ipInfo = ipInfo.replace("0|", "");

            }

            return ipInfo;

        } catch (Exception e) {

            e.printStackTrace();

        }

        return null;

    }

下面,我们编写main函数进行测试,发现可以正常的解析出ip信息

由于 ip 属地在国内的话,只会展示省份,而国外的话,只会展示国家。所以我们还需要对这个方法进行一下封装,得到获取 IP 属地的信息。

/**

 * 获取IP属地

 * @param ip

 * @return

 */

public static String getIpPossession(String ip) {

    String cityInfo = getCityInfo(ip);

    if (!StringUtils.isEmpty(cityInfo)) {

        cityInfo = cityInfo.replace("|", " ");

        String[] cityList = cityInfo.split(" ");

        if (cityList.length > 0) {

            // 国内的显示到具体的省

            if ("中国".equals(cityList[0])) {

                if (cityList.length > 1) {

                    return cityList[1];

                }

            }

            // 国外显示到国家

            return cityList[0];

        }

    }

    return "未知";

}

下面,我们在找一个 国外的IP测试一下效果。可以看到已经能够正常的显示IP属地信息了~

到这里如果获取用户的 IP 属地已经完成啦,如果想要了解关于更多ip2region的功能,欢迎访问其Github地址进行学习。

                                             项目地址

https://github.com/lionsoul2014/ip2region

码文不易,学到的小伙伴点一点关注哦,一键三连加收藏哦!!!

有不懂的小伙伴可以加下我的微信,我们私底下悄悄内卷~~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容