基于 CIDR 前缀计算和转换 IPv6 地址
Calculate and Convert IPv6 Address based on CIDR Prefix
我有一个带有 CIDR 前缀的 IPv6 地址。 Ex: 2001:0:3238:DFE1:0063::FEFB/32
我正在寻找一个函数,以便转换 IPv6 地址以匹配提供的 CIDR 前缀中的位数。这意味着,如果提供前缀 32,则 IP 地址应从左开始有 32 位,其余应为零。在上面的例子中,期望的输出应该是:
2001:0000:0000:0000:0000:0000:0000:0000/32
我确实有其他功能可以像这样缩短所需的 IP 2001::/32
CIDR 前缀的范围为 0-128。
这是我目前所拥有的,基于此 Post here,但我不确定它是否提供了所需的输出。有人可以看看并提供帮助吗?我在这里直接考虑 CIDR 表示吗?
public function getCIDRMatchedIP( $ipAddress )
{
// Split in address and prefix length
list($firstaddrstr, $prefixlen) = explode('/', $ipAddress);
// Parse the address into a binary string
$firstaddrbin = inet_pton($firstaddrstr);
// Convert the binary string to a string with hexadecimal characters
# unpack() can be replaced with bin2hex()
# unpack() is used for symmetry with pack() below
$firstaddrhex = reset(unpack('H*', $firstaddrbin));
// Overwriting first address string to make sure notation is optimal
$firstaddrstr = inet_ntop($firstaddrbin);
$cidrMatchedString = $firstaddrstr.'/'.$prefixlen;
echo $cidrMatchedString;
}
此处:
function getCIDRMatchedIP($ip)
{
if (strpos($ip, "::") !== false) {
$parts = explode(":", $ip);
$cnt = 0;
// Count number of parts with a number in it
for ($i = 0; $i < count($parts); $i++) {
if (is_numeric("0x" . $parts[$i]))
$cnt++;
}
// This is how many 0000 blocks is needed
$needed = 8 - $cnt;
$ip = str_replace("::", str_repeat(":0000", $needed), $ip);
}
list($ip, $prefix_len) = explode('/', $ip);
$parts = explode(":", $ip);
// This is the start bit mask
$bstring = str_repeat("1", $prefix_len) . str_repeat("0", 128 - $prefix_len);
$mins = str_split($bstring, 16);
$start = array();
for ($i = 0; $i < 8; $i++) {
$min = base_convert($mins[$i], 2, 16);
$start[] = sprintf("%04s", dechex(hexdec($parts[$i]) & hexdec($min)));
}
$start = implode(':', $start);
return $start . '/' . $prefix_len;
}
你调用函数 getCIDRMatchedIP
:
echo getCIDRMatchedIP('2001:0:3238:DFE1:0063::FEFB/32');
我有一个带有 CIDR 前缀的 IPv6 地址。 Ex: 2001:0:3238:DFE1:0063::FEFB/32
我正在寻找一个函数,以便转换 IPv6 地址以匹配提供的 CIDR 前缀中的位数。这意味着,如果提供前缀 32,则 IP 地址应从左开始有 32 位,其余应为零。在上面的例子中,期望的输出应该是:
2001:0000:0000:0000:0000:0000:0000:0000/32
我确实有其他功能可以像这样缩短所需的 IP 2001::/32
CIDR 前缀的范围为 0-128。
这是我目前所拥有的,基于此 Post here,但我不确定它是否提供了所需的输出。有人可以看看并提供帮助吗?我在这里直接考虑 CIDR 表示吗?
public function getCIDRMatchedIP( $ipAddress )
{
// Split in address and prefix length
list($firstaddrstr, $prefixlen) = explode('/', $ipAddress);
// Parse the address into a binary string
$firstaddrbin = inet_pton($firstaddrstr);
// Convert the binary string to a string with hexadecimal characters
# unpack() can be replaced with bin2hex()
# unpack() is used for symmetry with pack() below
$firstaddrhex = reset(unpack('H*', $firstaddrbin));
// Overwriting first address string to make sure notation is optimal
$firstaddrstr = inet_ntop($firstaddrbin);
$cidrMatchedString = $firstaddrstr.'/'.$prefixlen;
echo $cidrMatchedString;
}
此处:
function getCIDRMatchedIP($ip)
{
if (strpos($ip, "::") !== false) {
$parts = explode(":", $ip);
$cnt = 0;
// Count number of parts with a number in it
for ($i = 0; $i < count($parts); $i++) {
if (is_numeric("0x" . $parts[$i]))
$cnt++;
}
// This is how many 0000 blocks is needed
$needed = 8 - $cnt;
$ip = str_replace("::", str_repeat(":0000", $needed), $ip);
}
list($ip, $prefix_len) = explode('/', $ip);
$parts = explode(":", $ip);
// This is the start bit mask
$bstring = str_repeat("1", $prefix_len) . str_repeat("0", 128 - $prefix_len);
$mins = str_split($bstring, 16);
$start = array();
for ($i = 0; $i < 8; $i++) {
$min = base_convert($mins[$i], 2, 16);
$start[] = sprintf("%04s", dechex(hexdec($parts[$i]) & hexdec($min)));
}
$start = implode(':', $start);
return $start . '/' . $prefix_len;
}
你调用函数 getCIDRMatchedIP
:
echo getCIDRMatchedIP('2001:0:3238:DFE1:0063::FEFB/32');