我有一个庞大的 IP 地址动态列表、它们的子网掩码和一组给定的子网。我需要一些代码来识别每个地址所在的子网。我保持这个相当模糊,因为我对许多不同的解决方案持开放态度。有人在 C# 中有一些代码可以做类似的事情吗?谢谢!
问问题
2220 次
3 回答
2
首先,您需要代码将地址屏蔽到其子网:
public IPAddress ApplyMask(IPAddress address, IPAddress mask) {
byte[] addressBytes = address.GetAddressBytes();
byte[] maskBytes = mask.GetAddressBytes();
var appliedMaskBytes =
addressBytes.Zip(
maskBytes,
(addressByte, maskByte) => (byte)(addressByte & maskByte)
)
.ToArray();
return new IPAddress(appliedMaskBytes);
}
然后,假设你有一个Dictionary<IPAddress, IPAddress>
从 IP 地址到它的掩码:
IDictionary<IPAddress, IPAddress> maskDictionary;
var masked = ipAddresses
.Select(ipAddress =>
new { IPAddress = ipAddress,
Subnet = ApplyMask(
ipAddress,
maskDictionary[ipAddress]
)
}
)
.GroupBy(x => x.Subnet);
现在您有一个按子网划分的地址列表。相应地进行。
(注意:我手边没有编译器。对于轻微的编译器错误,我深表歉意。)
于 2011-09-13T14:18:29.770 回答
1
您不需要子网列表 - 可以推断它们。你可以用一些按位数学来做到这一点:
// Key is {Subnet, Subnet Mask}.
// Value is list of IPs in that range.
public static IDictionary<Tuple<IPAddress, IPAddress>, ISet<IPAddress>> AllocateToSubnets(IEnumerable<Tuple<IPAddress, IPAddress>> ips)
{
var dic = new Dictionary<Tuple<IPAddress, IPAddress>, ISet<IPAddress>>();
foreach (var item in ips)
{
var subnet = Tuple.Create(GetSubnetAddress(item.Item1, item.Item2), item.Item2);
ISet<IPAddress> set;
if (!dic.TryGetValue(subnet, out set))
dic.Add(subnet, set = new HashSet<IPAddress>());
if (!set.Add(item.Item1))
{
// Throw an error if you are looking for duplicates.
}
}
return dic;
}
private static IPAddress GetSubnetAddress(IPAddress address, IPAddress mask)
{
var b1 = address.GetAddressBytes();
var b2 = mask.GetAddressBytes();
if (b1.Length != b2.Length)
throw new InvalidOperationException("IPAddress and its mask do not share the same family.");
for (var i = 0; i < b1.Length; i++)
b1[i] &= b2[i];
return new IPAddress(b1);
}
测试代码:
var ips = new[]
{
Tuple.Create(IPAddress.Parse("192.168.0.1"), IPAddress.Parse("255.255.255.0")),
Tuple.Create(IPAddress.Parse("192.168.0.1"), IPAddress.Parse("255.255.0.0")),
Tuple.Create(IPAddress.Parse("192.168.0.2"), IPAddress.Parse("255.255.255.0")),
Tuple.Create(IPAddress.Parse("192.168.0.10"), IPAddress.Parse("255.255.255.0")),
Tuple.Create(IPAddress.Parse("192.168.0.10"), IPAddress.Parse("255.255.0.0")),
Tuple.Create(IPAddress.Parse("10.0.0.1"), IPAddress.Parse("255.255.255.0")),
Tuple.Create(IPAddress.Parse("10.1.0.1"), IPAddress.Parse("255.255.0.0")),
Tuple.Create(IPAddress.Parse("10.0.0.2"), IPAddress.Parse("255.255.255.0")),
Tuple.Create(IPAddress.Parse("10.1.0.10"), IPAddress.Parse("255.255.255.0")),
Tuple.Create(IPAddress.Parse("10.1.0.10"), IPAddress.Parse("255.255.0.0")),
};
var subnets = AllocateToSubnets(ips);
于 2011-09-13T14:30:54.937 回答
0
这取决于您的子网的定义。如果它是一个 IP 地址范围,您可以使用:使用 c# 根据一系列值检查 IP 地址的最简单方法
如果它是一个掩码,您可以使用它ip&(~mask)
来获取较低值并ip|(~mask)
获取范围并按上述方式进行操作。
如果子网给出了网络掩码的位数,您可以计算:
mask=~(UInt32.MaxValue>>n)
并按上述方式使用。
如果您想找出给定 IP 的网络掩码,那么我不知道。也不知道 IPv6。
于 2011-09-13T14:13:53.420 回答