文章目录
- 开始
- 问题
- 参考答案
- 答案解析
- 计算工具
- 测试
开始
好久没有看计算网络,感觉已经完全返给老师了。
最近,有同事遇到个问题,网络一直不对,又开始重新看一下。
相信很多朋友长时间不看也忘了,所以,这里记录一下,并提供了一个工具类用来计算相关值。
希望帮助新学习的朋友学习、已经忘了的朋友重新回忆。
觉得自己理解到位的朋友,可以尝试一下下面的问题。
问题
我的IPv4地址是:172.17.1.6,子网掩码是:255.255.252.0
请问:
- 我和172.17.0.6在同一网段吗?172.17.2.6呢?172.17.3.6呢?172.17.4.6呢?
- 我所在网络的网络地址是多少?网络位数是多少?
- 我所在网络的广播地址是多少?
- 我所在网络的最小主机IP地址是多少?
- 我所在的网络最大主机IP地址是多少?
上面的问题你能回答几个?答对了几个?
参考答案
- 172.17.1.6和172.17.0.6、172.17.2.6、172.17.3.6是同一网段,和172.17.4.6不是同一网段
- 我所在网络的网络地址:10101100000100010000000000000000(172.17.0.0/22)
- 我所在网络的广播地址:10101100000100010000001111111111(172.17.3.255)
- 我所在网络的最小主机IP地址:10101100000100010000000000000001(172.17.0.1)
- 我所在的网络最大主机IP地址:10101100000100010000001111111110(172.17.3.254)
答案解析
首先算网络地址:
我的IP & 子网掩码就是我所在的网络地址:
我的地址:10101100000100010000000100000110(172.17.1.6)
子网掩码:11111111111111111111110000000000(255.255.252.0)
与有0为0:10101100000100010000000000000000(172.17.0.0)
网络位数,数子网掩码前面1的个数:22,因此网络号可以写成:172.17.0.0/22,很多网策支持这样配置。
广播地址是主机位全为1的地址,因此把网络地址中的主机位全换为1即可:
网络位数是22,所以主机位是10位:10101100000100010000001111111111(172.17.3.255)
最小的主机地址是网络地址+1:10101100000100010000000000000001(172.17.0.1)
最大的主机地址是广播地址-1:10101100000100010000001111111110(172.17.3.254)
有了最小的主机地址和最大的主机地址,现在再看第一个问题,是不是清晰了。
这其中最大的误区就是,一晃眼就认为172.17.1.6的网段是172.17.1.0
这是没有CIDR,在子网掩码为:255.255.255.0时才成立
计算工具
package vip.meet.network.ip;
import lombok.extern.slf4j.Slf4j;
import java.math.BigInteger;
/**
* CIDR
* 网络中:
* 全为0的是:网络地址(最小地址)
* 全为1的是:广播地址(最大地址)
* 最小可用地址=网络地址+1
* 最大可用地址=广播地址-1
* <p>
* 私有地址:
* A类地址范围:10.0.0-10.255.255.255
* B类地址范围:172.16.0.0-172.31.255.555
* C类地址范围:192.168.0.0-192.168.255.255
*/
@Slf4j
public class IpCalculateHelper {
/**
* 根据主机IP和子网掩码计算网络地址 ip & mask
* <p>
* 198.1.33.205 :11000110000000010010000111001101
* 255.255.252.0:11111111111111111111110000000000
* &
* 198.1.32.0/22:11000110000000010010000000000000
*
* @param netIp 要计算的ip 198.1.33.205
* @param mask 子网掩码 255.255.252.0
* @return 网络地址 198.1.32.0
*/
public static String getIPV4CIDRNet(String netIp, String mask) {
String ipv4CIDRBinary = getIPV4CIDRBinary(netIp, mask);
return binaryIp2NetIp(ipv4CIDRBinary);
}
/**
* 获取网络二进制地址
*
* @param netIp 198.1.33.205
* @param mask 55.255.252.0
* @return 11000110000000010010000000000000
*/
public static String getIPV4CIDRBinary(String netIp, String mask) {
BigInteger ipBin = new BigInteger(getIPBinary(netIp), 2);
BigInteger maskBin = new BigInteger(getIPBinary(mask), 2);
BigInteger result = ipBin.and(maskBin);
String string = result.toString(2);
return "0".repeat(32 - string.length()) + string;
}
/**
* 二进制转ip转网络ip
*
* @param binaryIp 二进制ip 11000110000000010010000000000000
* @return ip 198.1.32.0
*/
public static String binaryIp2NetIp(String binaryIp) {
int length = binaryIp.length();
if (length > 32) {
throw new RuntimeException("非法ip长度:" + binaryIp);
}
String pad = "0".repeat(32 - length) + binaryIp;
return Integer.valueOf(pad.substring(0, 8), 2) + "." +
Integer.valueOf(pad.substring(8, 16), 2) + "." +
Integer.valueOf(pad.substring(16, 24), 2) + "." +
Integer.valueOf(pad.substring(24), 2);
}
/**
* 二进制ip转ip
*
* @param netIp 198.1.32.0
* @return binary ip 二进制ip 11000110000000010010000000000000
*/
public static String getIPBinary(String netIp) {
String[] parts = netIp.split("\\.");
if (parts.length != 4) {
throw new RuntimeException("非法ipv4:" + netIp);
}
StringBuilder sb = new StringBuilder();
for (String part : parts) {
BigInteger integer = new BigInteger(part, 10);
String partIp = integer.toString(2);
sb.append("0".repeat(8 - partIp.length())).append(partIp);
}
return sb.toString();
}
/**
* 根据子网掩码算网络位数
*
* @param mask 子网掩码 255.255.252.0
* @return 网络位数 22
*/
public static int getNetBitFromMask(String mask) {
String binString = getIPBinary(mask);
int count = 0;
for (int i = 0; i < binString.length(); i++) {
if (binString.charAt(i) != '1') {
break;
}
count++;
}
return count;
}
/**
* 获取二进制子网掩码
*
* @param netBit 网络位数
* @return 二进制子网掩码
*/
public static String getMaskBinaryFromNetBit(int netBit) {
if (netBit < 1 || netBit >= 32) {
throw new RuntimeException("非法网络位数:" + netBit);
}
return "1".repeat(netBit) + "0".repeat(32 - netBit);
}
/**
* 获取子网掩码
*
* @param netBit 网络位数
* @return 子网掩码
*/
public static String getMaskFromNetBit(int netBit) {
if (netBit < 1 || netBit >= 32) {
throw new RuntimeException("非法网络位数:" + netBit);
}
String maskBinary = "1".repeat(netBit) + "0".repeat(32 - netBit);
return binaryIp2NetIp(maskBinary);
}
/**
* 获取广播地址
*
* @param binaryCIDR cidr网络 11000110000000010010000000000000
* @param netBit 网络位数 22
* @return 广播地址 11000110000000010010001111111111
*/
public static String getBroadcast(String binaryCIDR, int netBit) {
if (netBit < 1 || netBit >= 32) {
throw new RuntimeException("非法网络位数:" + netBit);
}
return binaryCIDR.substring(0, netBit) + "1".repeat(32 - netBit);
}
/**
* 计算最小主机地址
*
* @param binaryCIDR CIDR网络地址 11000110000000010010000000000000
* @return 最小主机地址 1100011000000001001000000000000
*/
public static String minHostIp(String binaryCIDR) {
return binaryCIDR.substring(0, 31) + "1";
}
/**
* 最大主机地址
*
* @param binaryCIDR CIDR网络地址 11000110000000010010000000000000
* @param netBit 网络位数 22
* @return 最大主机地址 11000110000000010010001111111110
*/
public static String maxHostIp(String binaryCIDR, int netBit) {
if (netBit < 1 || netBit >= 32) {
throw new RuntimeException("非法网络位数:" + netBit);
}
return binaryCIDR.substring(0, netBit) + "1".repeat(31 - netBit) + "0";
}
public static void printNetInfo(String netIp, int netBit) {
if (netBit < 1 || netBit >= 32) {
throw new RuntimeException("非法网络位数:" + netBit);
}
String mask = getMaskFromNetBit(netBit);
printNetInfo(netIp, netBit, mask);
}
public static void printNetInfo(String netIp, String mask) {
int netBit = getNetBitFromMask(mask);
printNetInfo(netIp, netBit, mask);
}
public static void printNetInfo(String netIp, Integer netBit, String mask) {
if (netBit == null) {
netBit = getNetBitFromMask(mask);
}
if (netBit < 1 || netBit >= 32) {
throw new RuntimeException("非法网络位数:" + netBit);
}
String ipBinary = getIPBinary(netIp);
System.out.printf("二进制IP:%s(%s)\n", ipBinary, netIp);
String maskBinary = getIPBinary(mask);
System.out.printf(" 掩码IP:%s(%s)\n", maskBinary, mask);
String ipv4CIDRBinary = getIPV4CIDRBinary(netIp, mask);
System.out.printf(" 网络IP:%s(%s/%d)\n", ipv4CIDRBinary,
binaryIp2NetIp(ipv4CIDRBinary), netBit);
String broadcast = getBroadcast(ipv4CIDRBinary, netBit);
System.out.printf(" 广播IP:%s(%s)\n", broadcast, binaryIp2NetIp(broadcast));
String min = minHostIp(ipv4CIDRBinary);
System.out.printf(" 最小HIP:%s(%s)\n", min, binaryIp2NetIp(min));
String max = maxHostIp(ipv4CIDRBinary, netBit);
System.out.printf(" 最大HIP:%s(%s)\n", max, binaryIp2NetIp(max));
}
}
测试
@Test
public void printNetInfo() {
IpCalculateHelper.printNetInfo("10.2.2.7", 20);
System.out.println("--------------");
IpCalculateHelper.printNetInfo("172.17.1.6", "255.255.252.0");
System.out.println("--------------");
IpCalculateHelper.printNetInfo("192.168.3.3", 21);
}