您现在的位置是:首页 >其他 >ip转换相关知识详解网站首页其他

ip转换相关知识详解

BUG弄潮儿 2025-03-04 12:01:02
简介ip转换相关知识详解

判断某个ip是否在一个网段(ip+掩码)内

大学学的计算机网络已经忘了一大半,今天接到一个需求,需要判断用户的请求ip是否在办公网网段,如果是,那么就要做出相应的提示。

一开始我以为,办公网段就只需要截取前三位就行,比如:22.3.4.6 所在网段就是22.3.4就可以。

可是安全管理员给我发的却不是这样。而是以下这样的。

192.168.1.64/26

192.168.0.0/23

192.168.0.0/24

192.168.0.0/32

这些是什么意思呢?

前面的192.168.1.64表示ip后面的26表示掩码,就是表示192.168.1.64的前26位不能动,ip分为四部分,每一辈分占8位,一共32位,也就是说32位中有26位是不能动的,前面的192.168.1占8*3=24位不能动。后面的64是0100 0000 前面已经有24位不能动了,所以后面还有2位不能动就是01不能动。所以网段就是192.168.1.64-192.168.1.127

同理192.168.0.0/23就是前23位不能动,192.168不能动 8*2=16位 后面有7位不能动后面的0就是0000 0000其中前7位不能动,后一位随便是0或者1 所以网段是192.168.0-192.168.1

192.168.0.0/24前24位不能动,192.168.0不能动 8*3=24位 后面有8位随便动 所以网段是192.168.0

192.168.0.0/32是前32位都不能动所以网段是 192.168.0.0

package com.ip;

public class IpTest {
   public static void main(String[] args) {
       System.out.println(isInRange("192.168.1.127", "192.168.1.64/26"));
       System.out.println(isInRange("192.168.1.2", "192.168.0.0/23"));
       System.out.println(isInRange("192.168.0.1", "192.168.0.0/24"));
       System.out.println(isInRange("192.168.0.0", "192.168.0.0/32"));
   }
   public static boolean isInRange(String ip, String cidr) {
       String[] ips = ip.split("\.");
       int ipAddr = (Integer.parseInt(ips[0]) << 24)
               | (Integer.parseInt(ips[1]) << 16)
               | (Integer.parseInt(ips[2]) << 8) | Integer.parseInt(ips[3]);
       int type = Integer.parseInt(cidr.replaceAll(".*/", ""));
       int mask = 0xFFFFFFFF << (32 - type);
       String cidrIp = cidr.replaceAll("/.*", "");
       String[] cidrIps = cidrIp.split("\.");
       int cidrIpAddr = (Integer.parseInt(cidrIps[0]) << 24)
               | (Integer.parseInt(cidrIps[1]) << 16)
               | (Integer.parseInt(cidrIps[2]) << 8)
               | Integer.parseInt(cidrIps[3]);

       return (ipAddr & mask) == (cidrIpAddr & mask);
   }
}
https://blog.csdn.net/peng2hui1314/article/details/129027169
https://blog.csdn.net/u012045045/article/details/102514535

根据IP地址/掩码位(CIDR)和起始IP-终止IP计算网段(IPV4和IPV6)

计算IP列表

IPV4

  • 需要的jar包
			<dependency>
                <groupId>cn.hutool</groupId>
                <artifactId>hutool-all</artifactId>
                <version>5.7.7</version>
            </dependency>

代码示例

    /**
     * 获取IPV4 CIDR形式下所有的ipv4
     *
     * @param cidr IPV4的CIDR格式 (192.168.1.0/24)
     */
    private static R getIpsV4ByCidr(String cidr) {
        String ip = cidr.split("/")[0];
        int mask = Integer.parseInt(cidr.split("/")[1]);
        String ipFrom = Ipv4Util.getBeginIpStr(ip, mask);
        String ipTo = Ipv4Util.getEndIpStr(ip, mask);
        if (Ipv4Util.countByIpRange(ipFrom, ipTo) > 1024) {
            return R.warn("最多可以支持1024个IP");
        }
        return R.ok().addData(Ipv4Util.list(ip, mask, true));
    }

    /**
     * 获取IPV4 区间范围下所有的ipv4
     *
     * @param ipFrom IPV4的起始ip
     * @param ipTo   IPV4的结束ip
     */
    private static R getIpsV4ByRange(String ipFrom, String ipTo) {
        if (Ipv4Util.countByIpRange(ipFrom, ipTo) > 1024) {
            return R.warn("最多可以支持1024个IP");
        }
        return R.ok().addData(Ipv4Util.list(ipFrom, ipTo));
    }

IPV6

需要的jar包

		<dependency>
            <groupId>com.googlecode.java-ipv6</groupId>
            <artifactId>java-ipv6</artifactId>
            <version>0.17</version>
        </dependency>

代码示例

/**
     * 获取IPV6 CIDR形式下所有的ipv6
     *
     * @param cidr IPV6的CIDR格式 (fe80::2001/64)
     */
    private static R getIpsV6ByCidr(String cidr) {
        List<String> ipv6List = Lists.newArrayList();
        IPv6Network network = IPv6Network.fromString(cidr);
        if (network.size().compareTo(new BigInteger(String.valueOf(1024))) > 0) {
            return R.warn("最多可以支持1024个IP");
        }
        network.forEach(e -> ipv6List.add(e.toString()));
        return R.ok().addData(ipv6List);
    }

    /**
     * 获取IPV6 区间范围下所有的ipv6
     *
     * @param ipFrom IPV6的起始ip
     * @param ipTo   IPV6的结束ip
     */
    private static R getIpsV6ByRange(String ipFrom, String ipTo) {
        List<String> ipv6List = Lists.newArrayList();
        IPv6AddressRange ipv6Range = IPv6AddressRange.fromFirstAndLast(IPv6Address.fromString(ipFrom), IPv6Address.fromString(ipTo));
        if (ipv6Range.size().compareTo(new BigInteger(String.valueOf(1024))) > 0) {
            return R.warn("最多可以支持1024个IP");
        }
        ipv6Range.forEach(e -> ipv6List.add(e.toString()));
        return R.ok().addData(ipv6List);
    }

IP校验

IPV4

public static boolean fieldIPv4Valid(String matchParams) {
		String ipv4Format = "^((25[0-5]|2[0-4][0-9]|1[0-9]{2}|[0-9]{1,2})\.){3}(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[0-9]{1,2})$";
		boolean flag = matchParams.matches(ipv4Format);
		return flag;
	}

IPV6

public static boolean fieldIPv6Valid(String matchParams)  {
		String ipv6Format = "^\s*((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:)))(%.+)?\s*$";
		boolean flag = matchParams.matches(ipv6Format);
		return flag;
	}

https://blog.csdn.net/qq_41936784/article/details/121371597
风语者!平时喜欢研究各种技术,目前在从事后端开发工作,热爱生活、热爱工作。