我有一个包含 IPv4 地址和 /24 网络列表的文件。
IP 地址已经是唯一的并按所有四个八位位组排序。
我想使用 bash 将属于同一 /24 子网的地址分组(如果该子网中有多个地址),否则,我想保留列表中的 IP 地址。
例如:
10.0.0.1
10.0.0.2
10.20.0.0/24
10.30.10.1
10.30.10.2
192.168.0.1
192.168.1.5
192.168.30.0/24
192.168.50.3
期望的输出:
10.0.0.0/24
10.20.0.0/24
10.30.10.0/24
192.168.0.1
192.168.1.5
192.168.30.0/24
192.168.50.3
答案1
基本思想是使用每个/24
网络作为存储该网络上的子网数组的密钥。然后根据每个数组的长度生成输出。然而,这种方法有一个缺点,那就是它不支持流式处理。
仅供参考
AWK
awk '{
if (index($0, "/") > 0) { ip_map[$0] = "" }
else {
split($0, parts, ".")
network = parts[1]"."parts[2]"."parts[3]".0/24"
if (network in ip_map) { ip_map[network] = ip_map[network]" "$0 }
else { ip_map[network] = $0 }
}
}
END {
for (k in ip_map) {
if (ip_map[k] == "" || split(ip_map[k], arr) > 1) { print k }
else { print ip_map[k] }
}
}' | sort -n
巴什
#!/bin/bash
# or use `readarray -t` to read input from a file
ips=(
"10.0.0.1"
"10.0.0.2"
"10.0.0.3"
"10.20.0.0/24"
"10.30.10.1"
"10.30.10.2"
"192.168.0.1"
"192.168.1.5"
"192.168.30.0/24"
"192.168.50.3"
)
declare -A ip_map
for ip in "${ips[@]}"; do
if [[ "$ip" == *"/"* ]]; then
ip_map["$ip"]=null
else
network=$(echo "$ip" | { IFS="."; read a b c d; echo "$a.$b.$c.0/24"; })
if [[ ${ip_map[$network]+_} ]]; then
ip_map["$network"]+=" $ip"
else
ip_map["$network"]="$ip"
fi
fi
done
for k in "${!ip_map[@]}"; do
if [[ "${ip_map[$k]}" == null || $(echo "${ip_map[$k]}" | wc -w) -gt 1 ]]; then
echo "$k"
else
echo "${ip_map[$k]}"
fi
done | sort -n
Python
import ipaddress
ips = [
"10.0.0.1",
"10.0.0.2",
"10.0.0.3",
"10.20.0.0/24",
"10.30.10.1",
"10.30.10.2",
"192.168.0.1",
"192.168.1.5",
"192.168.30.0/24",
"192.168.50.3",
]
ip_map = {}
for ip in ips:
if "/" in ip:
ip_map[ip] = None
else:
network = ipaddress.ip_network(ip + "/24", strict=False)
try:
ip_map[network].append(ip)
except KeyError:
ip_map[network] = [ip]
for k, v in ip_map.items():
if v is None or len(v) > 1:
print(k)
elif len(v) == 1:
print(v[0])