Java 快速判断一个 IP 是否在给定的网段内

文章介绍了两种在Java中判断IP地址是否属于特定子网的方法:一是使用InetAddress类进行与操作;二是通过二进制计算实现。提供了详细的代码示例和解释。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

要在Java中判断一个IP地址是否在给定的网段内,可以使用子网掩码IP地址子网掩码进行与操作来提取网络地址,并将其与给定的子网地址进行比较。

方法一:借助于 Java 提供的 InetAddress

下面的例子由强大的 ChatGPT 提供

在这里插入图片描述

代码如下所示(子网掩码的计算可以截取字符串后,借助底部的算法进行获得):

public static boolean isIpAddressInSubnet(String ipAddress, String subnetAddress, String subnetMask) throws UnknownHostException {
    InetAddress inetAddress = InetAddress.getByName(ipAddress);
    InetAddress subnet = InetAddress.getByName(subnetAddress);
    InetAddress mask = InetAddress.getByName(subnetMask);

    byte[] inetAddressBytes = inetAddress.getAddress();
    byte[] subnetBytes = subnet.getAddress();
    byte[] maskBytes = mask.getAddress();

    for (int i = 0; i < inetAddressBytes.length; i++) {
        int addressByte = inetAddressBytes[i] & 0xFF;
        int subnetByte = subnetBytes[i] & 0xFF;
        int maskByte = maskBytes[i] & 0xFF;

        if ((addressByte & maskByte) != (subnetByte & maskByte)) {
            return false;
        }
    }

    return true;
}

这个方法接受三个参数:要检查的IP地址子网地址子网掩码。它使用InetAddress类将这些字符串转换为Java对象,然后将它们转换为字节数组。然后,它迭代每个字节,并将每个字节的值与相应的子网字节和掩码字节进行比较,以提取网络地址。如果网络地址与子网地址不匹配,则返回false,否则返回true

例如,假设我们要检查IP地址 “192.168.1.100” 是否在子网"192.168.1.0/24"中:

boolean result = isIpAddressInSubnet("192.168.1.100", "192.168.1.0", "255.255.255.0");
if (result) {
    System.out.println("IP address is in subnet");
} else {
    System.out.println("IP address is not in subnet");
}

方法二:撸个算法实现(二进制计算)

代码来源于 Google,但是明白原理即可,不需要深究。不管怎么样肯定都会需要 IP 地址子网掩码才能组成一个子网范围;然后判断另一个给定的 IP 是否在这个网段内。

/**
 * 判断是否在该网段中
 *
 * @param subnetRange 子网范围 x.x.x.x/xx 形式
 */
public static boolean isIpAddressInSubnet(String ipAddress, String subnetRange) {
    String[] networkips = ipAddress.split("\\.");
    int ipAddr = (Integer.parseInt(networkips[0]) << 24)
            | (Integer.parseInt(networkips[1]) << 16)
            | (Integer.parseInt(networkips[2]) << 8)
            | Integer.parseInt(networkips[3]);
            
    // 拿到主机数       
    int type = Integer.parseInt(subnetRange.replaceAll(".*/", ""));
    int ipCount = 0xFFFFFFFF << (32 - type);
    
    String maskIp = subnetRange.replaceAll("/.*", "");
    String[] maskIps = maskIp.split("\\.");
    
    int cidrIpAddr = (Integer.parseInt(maskIps[0]) << 24)
            | (Integer.parseInt(maskIps[1]) << 16)
            | (Integer.parseInt(maskIps[2]) << 8)
            | Integer.parseInt(maskIps[3]);
            
    return (ipAddr & ipCount) == (cidrIpAddr & ipCount);
}

例如,假设我们要检查IP地址 “192.168.1.100” 是否在子网"192.168.1.0/24"中:

boolean result = isIpAddressInSubnet("192.168.1.100", "192.168.1.0/24");
if (result) {
    System.out.println("IP address is in subnet");
} else {
    System.out.println("IP address is not in subnet");
}

其他

数字转为子网掩码

public static String subnetMaskFromPrefixLength(int prefixLength) {
    if (prefixLength < 0 || prefixLength > 32) {
        throw new IllegalArgumentException("Invalid prefix length");
    }

    int mask = 0xffffffff << (32 - prefixLength);
    return String.format("%d.%d.%d.%d",
        (mask & 0xff000000) >>> 24,
        (mask & 0x00ff0000) >>> 16,
        (mask & 0x0000ff00) >>> 8,
        (mask & 0x000000ff));
}

示例:

String subnetMask = subnetMaskFromPrefixLength(24);
System.out.println(subnetMask); // 输出:255.255.255.0

个人博客: Roc’s Blog

Java中检查两个IP地址是否属于同一个网络段,通常涉及到网络地址的计算和比较。你可以通过IP地址的掩码(subnet mask)来判断它们是否在同一网络。下面是一个简单的步骤: 1. **获取IP地址和掩码(子网掩码)**: 对于给定IPv4地址,你需要获取其对应的二进制形式,并计算出它的网络部分(包括前缀长度)和主机部分。 ```java import java.net.InetAddress; import java.util.regex.Pattern; public class IPUtils { private static final Pattern IP_PATTERN = Pattern.compile("^([0-9]{1,3}\\.){3}[0-9]{1,3}$"); public static boolean isSameNetwork(String ip1, String ip2, String subnet) throws Exception { if (!IP_PATTERN.matcher(ip1).matches() || !IP_PATTERN.matcher(ip2).matches()) { throw new IllegalArgumentException("Invalid IP address format"); } InetAddress addr1 = InetAddress.getByName(ip1); InetAddress addr2 = InetAddress.getByName(ip2); byte[] netmask = InetAddress.getByName(subnet).getHostAddress().getBytes(); int prefixLength = getPrefixLength(netmask); byte[] network1 = addr1.getAddress(); byte[] network2 = addr2.getAddress(); for (int i = 0; i < prefixLength; i++) { network1[i] &= netmask[i]; network2[i] &= netmask[i]; } return Arrays.equals(network1, network2); } private static int getPrefixLength(byte[] netmask) { int prefixLength = 0; while (prefixLength < 32 && netmask[prefixLength] == 255) { prefixLength++; } return prefixLength; } } ``` **如何使用:** ```java String existingIp = "192.168.1.1"; String newIp = "192.168.1.2"; String subnet = "255.255.255.0"; // 或者其他子网掩码 try { boolean inSameSegment = IPUtils.isSameNetwork(existingIp, newIp, subnet); System.out.println(inSameSegment ? "两个IP在同一个网段" : "不在同一个网段"); } catch (IllegalArgumentException | UnknownHostException e) { e.printStackTrace(); } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Roc.Chang

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值