当前位置: > 投稿>正文

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

04-09 互联网 未知 投稿

关于【java获取客户端ip】,java得到客户端ip地址吗,今天犇涌小编给您分享一下,如果对您有所帮助别忘了关注本站哦。

1、java获取客户端ip:全网显示 IP 归属地,用上这个开源库,实现也太简单了

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

动态显示IP属地

在蘑菇群聊中,也 可 以 展 示 IP 属 地,下面是小伙伴们在交流群中显示的

java获取客户端ip,java得到客户端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信息了

蘑菇最开始使用的是淘宝IP

地址:https://ip.taobao.com/

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

接入方式也比较简单,就是通过封装一个http 请求,传 入用户的 ip作为参数,就可以返回ip 对应的国家,省,城市 信息

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

原来的请求方式如下

/** * 获取IP地址来源 * * @param content 请求的参数 格式为:name=xxx&pwd=xxx * @param encodingString 服务器端请求编码。如GBK,UTF-8等 * @return * @throws UnsupportedEncodingException */public static String getAddresses(String content, String encodingString) { String ip = content.substring(3); if (!Util.isIpAddress(ip)) { log.info("IP地址为空"); return null; } // 淘宝IP宕机,目前使用Ip2region:https://github.com/lionsoul2014/ip2region String cityInfo = getCityInfo(ip); log.info("返回的IP信息:{}", cityInfo); // TODO 淘宝接口目前已经宕机,因此暂时注释下面代码 try { // 这里调用pconline的接口 String urlStr = "http://ip.taobao.com/service/getIpInfo.php"; // 从http://whois.pconline.com.cn取得IP所在的省市区信息 String returnStr = getResult(urlStr, content, encodingString); if (returnStr != null) { // 处理返回的省市区信息 log.info("调用IP解析接口返回的内容:" + returnStr); String[] temp = returnStr.split(","); //无效IP,局域网测试 if (temp.length < 3) { return "0"; } // 国家 String country = ""; // 区域 String area = ""; // 省 String region = ""; // 市 String city = ""; // 县 String county = ""; // 运营商 String isp = ""; Map<String, Object> map = JsonUtils.jsonToMap(returnStr); if (map.get("code") != null) { Map<String, String> data = (Map<String, String>) map.get("data"); country = data.get("country"); area = data.get("area"); region = data.get("region"); city = data.get("city"); county = data.get("area"); isp = data.get("isp"); } log.info("获取IP地址对应的地址" + country + "=" + area + "=" + region + "=" + city + "=" + county + "=" + isp); StringBuffer result = new StringBuffer(); result.append(country); result.append("|"); result.append(region); result.append("|"); result.append(city); result.append("|"); result.append(isp); return result.toString(); } } catch (Exception e) { log.error(e.getMessage()); return null; } return null;}

但是,之前接入淘宝IP库的时候,也经常会遇到服务不可用的情况,并且由于限制了QPS1,所以如果访问量大的话,就没办法获取了。

而到现在的话倒好了,这个接口也不对外提供服务了,直接下线了,不让调用了。

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

后面,陌溪在Github 冲浪的时候,发现 了 Ip2region项目。

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

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

数据聚合了一些知名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目录下

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

然后,通过内置的三种算法,分别转换用户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信息

java获取客户端ip,java得到客户端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属地信息了~

java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)

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

项目地址

https://github.com/lionsoul2014/ip2region

2、java获取客户端ip,java得到客户端ip地址吗

第二步:也就是此期教学的核心所在了,那究竟怎么获取客户端ip呢?我也就不卖关子了。

通常获取客户端的IP地址都是通过request.getRemoteAddr() ,对吧,但是你有想过么,如今基本系统都会进行域名代理等,比如通过 Apache,Squid 等反向代理软件,使用getRemoteAddr() 根本无法再获取到客户端的真实 IP 地址了。

为什么加了代理就获取不到了呢?这是因为在客户端和服务之间增加了中间代理,因此服务器是无法直接拿到客户端的 IP地址,服务器端应用也无法直接通过转发请求的地址返回给客户端,基本这种获取方式就直接被pass掉了。

如果还不理解,可以看下下边的代理流程示意图,你就会理解啦。

如果只局限系统不做代理,那肯定是ok的。那系统基本不可能不做代理的呀,所以咋办?别着急,我会告诉你怎么玩儿,如果使用了我的教学方法还获取不到,请你来揍我,好吧?

如下是具体获取客户端ip方法类:仅供大家参考。

/*** * 获取客户端ip地址 * @param request */ public static String getIP(final httpervletRequest request) throws Exception { if (request == null) { throw (new Exception("getIpAddr method httpervletRequest Object is null")); } String ipStr = request.getHeader("x-forwarded-for"); if (StringUtils.isBlank(ipStr) || "unknown".equalsIgnoreCase(ipStr)) { ipStr = request.getHeader("Proxy-Client-IP"); } if (StringUtils.isBlank(ipStr) || "unknown".equalsIgnoreCase(ipStr)) { ipStr = request.getHeader("WL-Proxy-Client-IP"); } if (StringUtils.isBlank(ipStr) || "unknown".equalsIgnoreCase(ipStr)) { ipStr = request.getRemoteAddr(); } // 多个路由时,取第一个非unknown的ip final String[] arr = ipStr.split(","); for (final String str : arr) { if (!"unknown".equalsIgnoreCase(str)) { ipStr = str; break; } } //目的是将localhost访问对应的ip 0:0:0:0:0:0:0:1 转成 127.0.0.1。 return ipStr.equals("0:0:0:0:0:0:0:1") ? "127.0.0.1" : ipStr; }复制代码

需要注意的是,自己本地开发,基本都是通过使用 [ localhost ]标准主机名来进行接口访问的,对吧,那么你就会特别好奇,数据库记录日志 ip 这一栏,ip有的竟然是 0:0:0:0:0:0:0:1。

为什么获取到的ip是这样呢?这是因为 0:0:0:0:0:0:0:1 是[ ipv6 ]的表现形式,对应[ ipv4 ]来说相当于127.0.0.1,也就是本机。所以我在最后通过三目运算强行把ipv6地址转成了ipv4地址。这样讲大家可否能理解?不理解就回头好好恶补一下计算机网络,这也不是不可以。

如下分别是通过 localhost 及实际ip分别进行接口访问,数据库保存的操作日志记录。

附上获取本地ip数据库截图:

本文关键词:java获取客户端ip和mac地址不同网段的,java获取客户端ip地址,java获取客户端地址,java获取客户端ip工具类,java获取客户端ip和端口。这就是关于《java获取客户端ip,java得到客户端ip地址吗(用上这个开源库)》的所有内容,希望对您能有所帮助!更多的知识请继续关注《犇涌向乾》百科知识网站:http://www.029ztxx.com!

版权声明: 本站仅提供信息存储空间服务,旨在传递更多信息,不拥有所有权,不承担相关法律责任,不代表本网赞同其观点和对其真实性负责。如因作品内容、版权和其它问题需要同本网联系的,请发送邮件至 举报,一经查实,本站将立刻删除。

猜你喜欢