百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

全网的 IP 归属地显示,带你5分钟加上,就是这么简单

off999 2025-01-17 12:34 19 浏览 0 评论

最近,继新浪微博之后,今日头条、腾讯、抖音、知乎、快手、小红书、百家号等各大平台陆陆续续都上线了"网络用户IP地址显示功能",境外用户显示的是国家,国内的用户显示的省份,而且此项显示无法关闭,归属地强制显示;

作为技术人,那!这个功能要怎么实现呢?

其实要想实现这个功能还是非常的容易,基于现成 GeoLite2离线库+免费的在线解析资源,5分钟就能整合了;

在整合之前,我们先简单了解一下,要想拿到用户的位置信息,有那些方式:

终端定位

我们的手机等电子设备都是带有GPS定位功能的,APP可以申请权限获取用户所处的经纬度坐标,根据坐标,就可以知道到用户所处的位置;比如百度、高德等地图厂商,就提供了完善的SDK,能非常方便的集成到应用,快速根据经纬度获取详细的位置详细;

优点

  • 快捷;
  • 准确;
  • 误差小。

缺点

  • 依赖硬件支持;
  • 依赖用户授权,如果用户不授权,APP将拿不到经纬度信息,导致失败;

IP地址解析

用户向服务端发起的请求都会带上IP地址,服务端拿到IP地址后,就能基于IP解析出用户的所处的位置;

优点

  • 无需授权,只要用户跟服务端交互,服务端就能拿到对应的IP信息

缺点

  • 准确性不高,位置可能存在偏差;
  • IP库更新不及时,导致部分IP归属地解析失败。

三方终端上报

比如,我们骑共享单车的时候,我们的位置信息就是通过单车的设备上报到服务器;

优点

  • 由三方终端基于GPS定位上报,不会获取个人设备的信息;
  • 准确快捷;
  • 专业设备,误差小;

缺点

  • 用户无法干预,信息会被迫强制上传至服务端,用户无法取消上传;

下面就来试着将 GeoLite2 免费 IP 库整合值SpringBoot项目,来获取用户的归属地信息;

1什么是GeoLite2?

GeoLite2数据库是免费的IP地理定位数据库;

优点:

  • 离线库,不需要网络
  • 数据库丰富
  • 速度快
  • 免费

缺点:

  • 准确度不高,存在偏差
  • 数据更新慢

2下载 GeoLite2 离线库

官网地址:https://www.maxmind.com/en/home

下载过程稍微有点点麻烦,这里下载了一份最新的,放在网盘,需要测试的可以直接通过这个链接下载:https://www.123pan.com/s/xPY9-J37vH

3SpringBoot 获取用户的IP

  • 工具类public class IpUtils {
    /**
    * 获取用户IP
    *
    @param request
    *
    @return
    */

    public static String getIpAddr(HttpServletRequest request) {

    String ip = request.getHeader("x-forwarded-for");

    if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
    ip = request.getHeader("X-Real-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.getRemoteAddr();
    }
    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_X_FORWARDED_FOR");
    }

    // 如果是多级代理,那么取第一个ip为客户ip
    if (ip != null && ip.indexOf(",") != -1) {
    ip = ip.substring(ip.lastIndexOf(",") + 1).trim();
    }
    return ip;
    }
    }
  • Controller获取HttpServletRequest通过上面的工具类,即可获取用户请求的真实IP;为了避免重复工作,这里也可以使用AOP解析出用户的IP信息,放到用户的请求对象中@RestController
    public class IpController {
    @GetMapping("/user/ip")
    public String userIp(HttpServletRequest request) {
    // 这里就能拿到用户的真实IP
    return IpUtils.getIpAddr(request);
    }
    }

4SpringBoot 整合 GeoLite2

  • 添加依赖<dependency>
    <groupId>com.maxmind.geoip2</groupId>
    <artifactId>geoip2</artifactId>
    <version>2.3.0</version>
    </dependency>

    <dependency>
    <groupId>com.maxmind.db</groupId>
    <artifactId>maxmind-db</artifactId>
    <version>1.0.0</version>
    </dependency>
  • 工具类public class GeoIpUtils {
    private static DatabaseReader reader;

    private static void init() {
    try {
    // 创建 GeoLite2 数据库 Reader
    // 这里可以放在本地磁盘,也可以随项目放在resource目录下
    File database = new File("F:\\web\\GeoLite2-City.mmdb");
    // 读取数据库内容
    reader = new DatabaseReader.Builder(database).build();
    } catch (Exception ex) {

    }
    }

    public static void getCityByIP(String ip) throws Exception {
    if (null == reader) {
    init();
    }

    InetAddress ipAddress = InetAddress.getByName(ip);

    // 获取查询结果
    CityResponse response = reader.city(ipAddress);

    // 获取国家信息
    Country country = response.getCountry();
    System.out.println("国家信息:" + JSON.toJSONString(country));

    // 获取省份
    Subdivision subdivision = response.getMostSpecificSubdivision();
    System.out.println("省份信息:" + JSON.toJSONString(subdivision));

    //城市
    City city = response.getCity();
    System.out.println("城市信息:" + JSON.toJSONString(city));

    // 获取城市
    Location location = response.getLocation();
    System.out.println("经纬度信息:" + JSON.toJSONString(location));
    }
    }
  • 测试public static void main(String[] args) throws Exception {
    String ip = "183.19.xxx.138";
    GeoIpUtils.getCityByIP(ip);
    }
    输出结果:国家信息:{"geoNameId":1814991,"isoCode":"CN","name":"China","names":{"de":"China","ru":"Китай","pt-BR":"China","ja":"中国","en":"China","fr":"Chine","zh-CN":"中国","es":"China"}}
    省份信息:{"geoNameId":1809935,"isoCode":"GD","name":"Guangdong","names":{"en":"Guangdong","fr":"Province de Guangdong","zh-CN":"广东"}}
    城市信息:{"geoNameId":1998011,"name":"Yanqianlaocun","names":{"en":"Yanqianlaocun","zh-CN":"岩前老村"}}
    经纬度信息:{"accuracyRadius":500,"latitude":23.3255,"longitude":116.5007,"timeZone":"Asia/Shanghai"}

就这么简单,轻轻松松就能拿到用户IP所处的国家、省份、城市、经纬度等详细信息,可以根据自己的业务需要,对这些数据再做进一步的封装。

5GeoLite2的其他用法

上面介绍的时SpringBoot整合GeoLite2,同样在其他的一些场景下,也是可以利用GeoLite2获取归属地信息;

  • 整合至Nignx,获取用户归属地信息Nginx 整合 GeoLite2 来解析用户的归属地信息,在代理层就直接整理好对应的数据;
  • ELK中整合GeoLite2ELK 日志整理的时候,可以通过GeoLite2 获取用户的IP归属地信息;然后通过Kibana,就能非常直观的展示用户的地域分布情况;ELK搭建,这才是看日志的正确姿势

6在线方案

上面一开始介绍GeoLite2时就列举了其离线库更新收录不及时的问题,可能导致一些IP在离线库中并不存在,查找的时候,就会报AddressNotFoundException的错误,如下示例:

遇到这种请求,我们要怎么办呢?

下面就来介绍几种在线IP归属地获取的方式,当本地离线库无法获取的时候,就可以利用三方的在线库,来补充完善;

在线获取的优点:

  • IP更新及时
  • 准确度高

缺点

  • 三方依赖性强
  • 需要付费,免费版本一般都有各种限制

以下示例中的xxx.xxx.xxx.xxx均代表ip地址;

百度

地址:https://opendata.baidu.com/api.php?query=xxx.xxx.xxx.xxx&resource_id=6006&co=&oe=utf8

响应数据:

{
  "status": "0",
  "t": "",
  "set_cache_time": "",
  "data": [
    {
      "ExtendedLocation": "",
      "OriginQuery": "183.19.xxx.138",
      "appinfo": "",
      "disp_type": 0,
      "fetchkey": "183.19.xxx.138",
      "location": "广东省肇庆市 电信",
      "origip": "183.19.xxx.138",
      "origipquery": "183.19.xxx.138",
      "resourceid": "6006",
      "role_id": 0,
      "shareImage": 1,
      "showLikeShare": 1,
      "showlamp": "1",
      "titlecont": "IP地址查询",
      "tplt": "ip"
    }
  ]
}

status等于0表示成功,1表示失败;可能存在status等于0,但是data中没有数据的情况。

ip-api接口

  • 本机的IP信息http://ip-api.com/json/
  • 指定国际化http://ip-api.com/json/?lang=zh-CN
  • 指定IP查询http://ip-api.com/json/xxx.xxx.xxx.xxx?lang=zh-CN返回数据:{
    "status": "success",
    "country": "中国",
    "countryCode": "CN",
    "region": "GD",
    "regionName": "广东",
    "city": "岩前老村",
    "zip": "",
    "lat": 23.3255,
    "lon": 116.5007,
    "timezone": "Asia/Shanghai",
    "isp": "Chinanet",
    "org": "Chinanet GD",
    "as": "AS4134 CHINANET-BACKBONE",
    "query": "183.19.xxx.138"
    }

搜狐IP查询

地址:http://pv.sohu.com/cityjson?ie=utf-8

返回数据比较的简单:

var returnCitySN = {"cip": "xxx.xxx.xxx.xxx", "cid": "440300", "cname": "广东省深圳市"};

太平洋IP地址查询

地址:http://whois.pconline.com.cn/ipJson.jsp?ip=xxx.xxx.xxx.xxx&json=true

返回数据:

{
  "ip": "183.17.xxx.138",
  "pro": "广东省",
  "proCode": "440000",
  "city": "深圳市",
  "cityCode": "440300",
  "region": "",
  "regionCode": "0",
  "addr": "广东省深圳市 电信",
  "regionNames": "",
  "err": ""
}

淘宝API接口

http://ip.taobao.com/service/getIpInfo.php?ip=xxx.xxx.xxx.xxx

{
    "code": 0,
    "data": {
        "ip": "183.17.xxx.138",
        "country": "中国",
        "area": "",
        "region": "广东",
        "city": "深圳",
        "county": "XX",
        "isp": "电信"
    }
}

code等于0表示成功,1表示失败

126

地址:https://ip.ws.126.net/ipquery?ip=xxx.xxx.xxx.xxx

响应数据:

var lo="广东省", lc="肇庆市"; 
var localAddress={city:"肇庆市", province:"广东省"}

响应的数据比较的简单

IP信息

地址:https://ip.useragentinfo.com/json?ip=xxx.xxx.xxx.xxx

响应数据:

{
  "country": "中国",
  "short_name": "CN",
  "province": "广东省",
  "city": "肇庆市",
  "area": "德庆县",
  "isp": "电信",
  "net": "",
  "ip": "183.19.xxx.138",
  "code": 200,
  "desc": "success"
}

这么多的姿势,实现起来是不是就非常的容易了;如果你对IP解析的需求比较依赖,也完全可以通过离线加这么多在线的方式,开发一个单独的IP解析模块,作为公司的基础服务,提供给内部其他模块使用。

示例目录:https://github.com/vehang/ehang-spring-boot/tree/main/spring-boot-004-request-validate/src/main/java/com/ehang/validate/geoip


来源:公众号—— 一行Java

相关推荐

安装python语言,运行你的第一行代码

#01安装Python访问Python官方(https://www.python.org/),下载并安装最新版本的Python。确保安装过程中勾选“Addpython.exetoPAT...

Python推导式家族深度解析:字典/集合/生成器的艺术

一、为什么需要其他推导式?当你在处理数据时:o需要快速去重→集合推导式o要建立键值映射→字典推导式o处理海量数据→生成器表达式这些场景是列表推导式无法完美解决的,就像工具箱需要不同工...

别再用循环创建字典了!Python推导式让你的代码起飞

当同事还在用for循环吭哧吭哧创建字典时,我早已用推导式完成3个需求了!这个被90%新手忽视的语法,今天让你彻底掌握字典推导式的4大高阶玩法,文末彩蛋教你用1行代码搞定复杂数据转换!基础语法拆解#传...

什么是Python中的生成器推导式?(python生成器的好处)

编程派微信号:codingpy本文作者为NedBatchelder,是一名资深Python工程师,目前就职于在线教育网站Edx。文中蓝色下划线部分可“阅读原文”后点击。Python中有一种紧凑的语法...

Python 列表转换为字符串:实用指南

为什么在Python中将列表转换为字符串?Python列表非常灵活,但它们并非在所有地方都适用。有时你需要以人类可读的格式呈现数据——比如在UI中显示标签或将项目保存到CSV文件。可能还...

生成器表达式和列表推导式(生成器表达式的计算结果)

迭代器的输出有两个很常见的使用方式,1)对每一个元素执行操作,2)选择一个符合条件的元素子集。比如,给定一个字符串列表,你可能想去掉每个字符串尾部的空白字符,或是选出所有包含给定子串的字符串。列表...

python学习——038python中for循环VS列表推导式

在Python中,for循环和列表推导式(ListComprehension)都可以用于创建和处理列表,但它们的语法、性能和适用场景有所不同。以下是两者的详细对比:1.语法结构for循环使用...

python中列表推导式怎么用?(列表 python)

这个问题,我们不妨用近期很火的ChatGPT来试试,来看看人工智能是如何解答的?在Python中,列表解析是一种简洁的方法,用于生成列表。它是一种快速,简洁的方法,可以在一行代码中生成列表,而不需...

Python列表推导式:让你的代码优雅如诗!

每次写for循环都要三四行代码?处理数据时总被嵌套结构绕晕?学会列表推导式,一行代码就能让代码简洁十倍!今天带你解锁这个Python程序员装(偷)逼(懒)神器!一、为什么你需要列表推导式?代码...

python学习——038如何将for循环改写成列表推导式

在Python里,列表推导式是一种能够简洁生成列表的表达式,可用于替换普通的for循环。下面是列表推导式的基本语法和常见应用场景。基本语法result=[]foriteminite...

太牛了!Python 列表推导式,超级总结!这分析总结也太到位了!

Python列表推导式,超级总结!一、基本概念列表推导式是Python中创建列表的一种简洁语法,它允许你在一行代码内生成列表,替代传统的for循环方式。其核心思想是**"对可迭代对...

25-2-Python网络编程-TCP 编程示例

2-TCP编程示例应用程序通常通过“套接字”(socket)向网络发出请求或者应答网络请求,使主机间或者一台计算机上的进程间可以通信。Python语言提供了两种访问网络服务的功能。其中低级别的网络服...

python编程的基础与进阶(周兴富)(python编程基础视频)

前不久我发文:《懂了,if__name=='__main__'》。想不到的是,这个被朋友称之为“读晕了”的文章,其收藏量数百,有效阅读量竟然过万。所谓“有效阅读量”,就是读到尾部才退...

Python 闭包:深入理解函数式编程的核心概念

一、简介在Python编程领域,闭包(Closure)是一个既基础又强大的概念,它不仅是装饰器、回调函数等高级特性的实现基础,更是函数式编程思想的重要体现。理解闭包的工作原理,能够帮助开发者编写出...

Python小白逆袭!7天吃透PyQt6,独立开发超酷桌面应用

PythonGUI编程:PyQt6从入门到实战的全面指南在Python的庞大生态系统中,PyQt6作为一款强大的GUI(GraphicalUserInterface,图形用户界面)编程框架,为开...

取消回复欢迎 发表评论: