2009-06-24 3 views
14

Я немного искал, чтобы найти код C# для преобразования сети в нотации CIDR (72.20.10.0/24) в диапазон IP-адресов, без большой удачи. Есть несколько потоков о CIDR на stackoverlow, но ни один из них не имеет кода на C# и не охватывает именно то, что мне нужно. Поэтому я решил приготовить его сам, и я не хотел, чтобы код полагался на System.Net для любых преобразований в этой версии.Как конвертировать CIDR в сеть и диапазон IP-адресов в C#?

Возможно, это может помочь кому-то.

Список литературы:

What's the best way to convert from network bitcount to netmask?

"Whatmask" С код из http://www.laffeycomputer.com/whatmask.html

Использование:

uint startIP, endIP; 
Network2IpRange("72.20.10.0/24", out startIP, out endIP); 

код принимает 32 бита для всего.

static void Network2IpRange(string sNetwork, out uint startIP, out uint endIP) 
{   
    uint ip,  /* ip address */ 
     mask,  /* subnet mask */    
     broadcast, /* Broadcast address */ 
     network; /* Network address */ 

    int bits;    

    string[] elements = sNetwork.Split(new Char[] { '/' });   

    ip = IP2Int(elements[0]); 
    bits = Convert.ToInt32(elements[1]); 

    mask = ~(0xffffffff >> bits); 

    network = ip & mask; 
    broadcast = network + ~mask; 

    usableIps = (bits >30)?0:(broadcast - network - 1); 

    if (usableIps <= 0) 
    { 
     startIP = endIP = 0; 
    } 
    else 
    { 
     startIP = network + 1;    
     endIP = broadcast - 1; 
    } 
} 

public static uint IP2Int(string IPNumber) 
{ 
    uint ip = 0; 
    string[] elements = IPNumber.Split(new Char[] { '.' }); 
    if (elements.Length==4) 
    { 
     ip = Convert.ToUInt32(elements[0])<<24; 
     ip += Convert.ToUInt32(elements[1])<<16; 
     ip += Convert.ToUInt32(elements[2])<<8; 
     ip += Convert.ToUInt32(elements[3]); 
    } 
    return ip; 
} 

Не стесняйтесь представить свои улучшения.

+0

Это было полезно для меня, спасибо. –

ответ

3

шаги будут идти, как это для network/maskBits,

Вы вычислить mask в одном из этих двух способов,

mask = ~((1 << (32 - maskBits)) - 1) // or, 
mask = ~(0xFFFFFFFF >> maskBits) 

то диапазон,

StartIP = network 
EndIP = network | ~mask 

Более точно ,

StartIP = network & mask 
EndIP = (network & mask) | ~mask 

Где,

  • << является побитовое сдвиг влево (без переворачивания)
  • & является побитовое И,
  • | является побитовое ИЛИ и
  • ~ является побитовое ОБРАТН.ПОТОЛОК.
+0

'mask = ~ (0xFFFFFFFF >> maskBits)' не будет выдавать правильное значение, если 'maskBits' равно 32 - значение будет 0x0 вместо 0xFFFFFFFF –

+0

На самом деле я не получил ни одного из этих методов для работы. '~ (0xFFFFFFFF >> 24)' возвращает '0.255.255.255' вместо' 255.255.255.0'. Что для меня работало: '0xFFFFFFFF >> -24' – cogumel0

6

Вот как вы это делаете для вашего примера 72.20.10.0/24,

Пусть Network быть 72.20.10.0
Mask является ~((1 << (32-24)) - 1) // или
Mask является ~(0xFFFFFFFF >> 24)

  • который 0xFFFFFF00

StartIP is - (Network & Mask);

  • который 72.20.10.0 & 0xFFFFFF00

EndIP это - ((Network & Mask) | ~Mask);

  • который (72.20.10.0 & 0xFFFFFF00) | 0x000000FF

Это будет 72.20.10.0 -- 72.20.10.255.

17

Я рекомендую использовать класс IP-сети C# от Github.

string net = "192.168.168.100/24"; 
IPNetwork ipnetwork = IPNetwork.Parse(net); 

Console.WriteLine("Network : {0}", ipnetwork.Network); 
Console.WriteLine("Netmask : {0}", ipnetwork.Netmask); 
Console.WriteLine("Broadcast : {0}", ipnetwork.Broadcast); 
Console.WriteLine("FirstUsable : {0}", ipnetwork.FirstUsable); 
Console.WriteLine("LastUsable : {0}", ipnetwork.LastUsable); 
Console.WriteLine("Usable : {0}", ipnetwork.Usable); 
Console.WriteLine("Cidr : {0}", ipnetwork.Cidr); 

Это Ouput

Network : 192.168.168.0 
Netmask : 255.255.255.0 
Broadcast : 192.168.168.255 
FirstUsable : 192.168.168.1 
LastUsable : 192.168.168.254 
Usable : 254 
Cidr : 24 

Удачи.

2

Вот как преобразовать CIDR нотации в диапазоне в T-SQL, from my blog post:

Сначала предварительно создать эту функцию в SQL Server (от http://www.stardeveloper.com).

CREATE FUNCTION [dbo].[ConvertIPToLong](@IP varchar(15)) 
RETURNS bigint 
AS 
BEGIN 
    DECLARE @Long bigint 
    SET @Long = CONVERT(bigint, PARSENAME(@IP, 4)) * 256 * 256 * 256 + 
     CONVERT(bigint, PARSENAME(@IP, 3)) * 256 * 256 + 
     CONVERT(bigint, PARSENAME(@IP, 2)) * 256 + 
     CONVERT(bigint, PARSENAME(@IP, 1)) 

    RETURN (@Long) 
END 

Это пример кода T-SQL Я поставил вместе, который будет вычислять низкого и высокого IP-диапазонов от адреса CIDR. Это грязно, и мне пришлось работать с отсутствием операторов смены битов T-SQL.

Declare @CidrIP varchar(50) 
Set @CidrIP = '10.100.60.55/28' 

Select dbo.[ConvertIPToLong](left(@CidrIP, patindex('%/%' , @CidrIP) - 1)) & (cast(4294967295 as bigint)^(Power(2, 32 - Cast(substring(@CidrIP, patindex('%/%' , @CidrIP) + 1, 2) as int)) - 1)) as LowRange, 
     dbo.[ConvertIPToLong](left(@CidrIP, patindex('%/%' , @CidrIP) - 1)) & (cast(4294967295 as bigint)^(Power(2, 32 - Cast(substring(@CidrIP, patindex('%/%' , @CidrIP) + 1, 2) as int)) - 1)) + (Power(2, 32 - Cast(substring(@CidrIP, patindex('%/%' , @CidrIP) + 1, 2) as int)) - 1) 
Смежные вопросы