根据IP判断地理位置

根据IP判断地理位置

  • 首先,当然是要有IP数据库,指明某一段IP代表哪个地方,本文采用的是txt格式的数据文件
  • IP数据库中的IP全部转换成了Long型,并且经过排序,这是为了方便二分查找

初始化操作: 载入IP数据文件

将数据文件按行读入内存,IP数据文件的格式如下:

起始IP结束IP国家未知运营商
1000000020000000中国北京市北京市海淀区未知电信

读取文件:

ipFile = new File(Class.forName("IPMapping").getClassLoader().getResource("ip_standard.txt").getFile());

List<String> lines = null;
lines = Files.readLines(ipFile, Charsets.UTF_8);

遍历每一行信息后存放在ImmutableList中:

static ImmutableList<Long> START_IP_INDEX;
static ImmutableList<Long> END_IP_INDEX;
static ImmutableList<String> IP_INFO;

START_IP_INDEX = new ImmutableList.Builder<Long>().addAll(startIpIndexTmp.iterator()).build();
END_IP_INDEX = new ImmutableList.Builder<Long>().addAll(endIpIndexTmp.iterator()).build();
IP_INFO = new ImmutableList.Builder<String>().addAll(ipInfoTmp.iterator()).build();

判断IP的准确性

需要查找的IP为标准的XXX.XXX.XXX.XXX格式
在进行查找之前需要判断该IP字符串是否有效:

public static boolean isValid(String ipStr) {
    if (StringUtils.isBlank(ipStr))
        return false;

    final String IP_FORMAT = "((25[0-5]|2[0-4]\\d|1\\d{2}|[1-9]\\d|\\d)\\.){3}(25[0-5]|2[0-4]\\d|1\\d{2}|[1-9]\\d|\\d)";

    return (ipStr.matches(IP_FORMAT)) ? true : false;
}

有效的IP字符串可以转化成Long型

public static long ip2long(String ipStr) {
    Long ipInLong = 0L;
    if (isValid(ipStr)) {
        String[] partIp = StringUtils.split(ipStr.trim(), "\\.");
        for (int i = 0; i < 4; i++)
            ipInLong = ipInLong << 8 | Integer.parseInt(partIp[i]);
    }
    return ipInLong;
}

通过二分查找判断该IP是否有对应的信息

和普通的二分相比,键的值是一对数,而且这对数的左值小于等于右值
改下了二分查找如下:

public int binarySearch(long ip) {

    int low = 0;
    int high = START_IP_INDEX.size() - 1;
    while (low <= high) {
        int middle = (low + high) / 2;
        if (ip >= START_IP_INDEX.get(middle) && (ip <= END_IP_INDEX.get(middle)))
            return middle;
        if (ip < START_IP_INDEX.get(middle))
            high = middle - 1;
        else
            low = middle + 1;
    }
    return -1;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值