Преобразование адреса CIDR в маску подсети и сетевой адрес

Учитывая адрес CIDR, например 192.168.10.0/24

  • Как определить длину маски? (24)
  • Как определить адрес по маске? (255.255.255.0)
  • Как определить сетевой адрес? (192.168.10.0)

person Dmitry    schedule 31.05.2010    source источник


Ответы (11)


Это покрыто apache utils.

См. Этот URL: http://commons.apache.org/proper/commons-net/apidocs/org/apache/commons/net/util/SubnetUtils.html.

String subnet = "192.168.0.3/31";
SubnetUtils utils = new SubnetUtils(subnet);

utils.getInfo().isInRange(address)

Примечание. Для использования, например, 32 подсетей CIDR, необходимо добавить следующее объявление:

utils.setInclusiveHostCount(true);
person Yuriy    schedule 28.07.2011
comment
похоже, у утилиты есть проблема с уведомлением только об одном адресе. Я имею в виду; допустим, IP - это всего лишь один IP. поэтому, если я хочу использовать его как ip / 32, он не работает. или я неправильно его использую. - person Olgun Kaya; 28.12.2011
comment
@OlgunKaya, вы используете его неправильно, конструктор, показанный в ответе, ожидает нотации CIDR, это IP_address/MASK, а не только IP_address. - person logoff; 26.03.2014
comment
getNetmask () возвращает маску подсети (255.255.255.0). Если я хочу получить длину префикса (24), есть ли какой-нибудь метод? Я не смог найти - person Nagabhushan S N; 19.06.2018
comment
SubnetUtils не поддерживает IPv6, как добиться того же для ipv6? - person Sayantan Ghosh; 06.05.2019

Вот как вы бы сделали это на Java,

    String[] parts = addr.split("/");
    String ip = parts[0];
    int prefix;
    if (parts.length < 2) {
        prefix = 0;
    } else {
        prefix = Integer.parseInt(parts[1]);
    }
    int mask = 0xffffffff << (32 - prefix);
    System.out.println("Prefix=" + prefix);
    System.out.println("Address=" + ip);

    int value = mask;
    byte[] bytes = new byte[]{ 
            (byte)(value >>> 24), (byte)(value >> 16 & 0xff), (byte)(value >> 8 & 0xff), (byte)(value & 0xff) };

    InetAddress netAddr = InetAddress.getByAddress(bytes);
    System.out.println("Mask=" + netAddr.getHostAddress());
person ZZ Coder    schedule 31.05.2010
comment
/ *** приближение в javascript. так как мне это было нужно для себя, я могу также поделиться *** / function cidrToMask (cidrStr) {var parts = cidrStr.split ('/'); var ipStr = parts [0]; var prefix = (parts.length ‹1)? 0: Число (части [1]); var mask = 0xffffffff ‹< (32 - префикс); var maskStr = [(маска ››› 24), (маска ›› 16 & 0xff), (маска ›› 8 & 0xff), (маска & 0xff)] .join ('.'); предупреждение (Префикс = + префикс + \ n + Адрес = + ipStr + \ n + Маска = + maskStr); }; cidrToMask ('192.168.10.0/24'); - person username; 07.11.2011

Библиотека Java IPAddress поддерживает как IPv4, так и IPv6 полиморфным способом, включая подсети. Документ javadoc доступен по ссылке. Отказ от ответственности: я руководитель проекта.

Все перечисленные вами варианты использования прозрачно поддерживаются как для IPv4, так и для IPv6.

    String str = "192.168.10.0/24";
    IPAddressString addrString = new IPAddressString(str);
    try {
         IPAddress addr = addrString.toAddress();
         Integer prefix = addr.getNetworkPrefixLength(); //24
         IPAddress mask = addr.getNetwork().getNetworkMask(prefix, false);//255.255.255.0
         IPAddress networkAddr = addr.mask(mask);  //192.168.10.0
         IPAddress networkAddrOtherWay = addr.getLower().removePrefixLength(); //192.168.10.0

         ...
    } catch(AddressStringException e) {
        //e.getMessage provides validation issue
    }
person Sean F    schedule 17.10.2016
comment
Доступно ли это как зависимость от Maven? - person Jeff Vincent; 28.10.2016
comment
скоро будет, обновлю страницу проекта, когда она - person Sean F; 29.10.2016
comment
теперь это так, теперь оно находится в центральном репо maven: repo1 .maven.org / maven2 / com / github / seancfoley / ipaddress / 2.0.0. - person Sean F; 05.03.2017
comment
Я читал документы, я экспериментировал ... есть ли простой способ получить, например, массив всех IP-адресов, которые можно проверить, скажем ... из вашего класса подсети? Или мне нужно заново строить адреса из нижнего и верхнего адреса? - person Michael Sims; 19.07.2018
comment
Что ж, вы можете определить, доступен ли адрес для проверки связи, только получив этот адрес и выполнив его проверку связи. Так что вам придется попробовать пинговать каждый из них. Чтобы получить каждый адрес, вы можете просто перебрать подсеть: for (IPAddress addr: new IPAddressString (192.168.10.0/24) .getAddress ()) {...} - person Sean F; 20.07.2018
comment
@SeanF Я надеялся, что там есть какой-нибудь класс модного танца, который сделает всю работу за меня ... ;-) - person Michael Sims; 30.04.2020
comment
@MichaelSims и ваши надежды оправдались :-) - person Sean F; 14.05.2020
comment
@SeanF На самом деле, мне пришлось сделать это так: for (IPAddress addr: new IPAddressString (10.10.10.0/24) .toSequentialRange (). GetIterable ()) {...} - Но это сработало, за исключением того, что мне пришлось выяснить, как изолировать адрес .0 от моего .isReachable, потому что Java говорит, что у меня нет разрешения на проверку связи с сетевым адресом ... иди разберись :-) - person Michael Sims; 03.06.2020

После ответа Юрия: Чтобы получить весь диапазон IP-адресов, Java-класс Apache SubnetUtils предлагает следующие методы:

String[] addresses = utils.getInfo().getAllAddresses();

Чтобы загрузить банку, содержащую класс, перейдите по ссылке: http://repo1.maven.org/maven2/commons-net/commons-net/3.0.1/commons-net-3.0.1.jar

Исходный код: http://svn.apache.org/viewvc/commons/proper/net/trunk/src/main/java/org/apache/commons/net/util/SubnetUtils.java?view=разметка

Идентификатор Maven:

<groupId>commons-net<groupId>
<artifactId>commons-net<artifactId>
<version>3.0.1<version>
person Andreas Panagiotidis    schedule 29.09.2011

Командная строка Linux ipcalc. Вы можете быстро использовать:

$ipcalc 192.168.10.0/24
Address:   192.168.10.0         11000000.10101000.00001010. 00000000
Netmask:   255.255.255.0 = 24   11111111.11111111.11111111. 00000000
Wildcard:  0.0.0.255            00000000.00000000.00000000. 11111111
=>
Network:   192.168.10.0/24      11000000.10101000.00001010. 00000000
HostMin:   192.168.10.1         11000000.10101000.00001010. 00000001
HostMax:   192.168.10.254       11000000.10101000.00001010. 11111110
Broadcast: 192.168.10.255       11000000.10101000.00001010. 11111111
Hosts/Net: 254                   Class C, Private Internet
person Trantkat Troy    schedule 02.05.2017
comment
macos: варить установить ipcalc - person Ahmed Rezk; 27.04.2020

Алгоритм написан в псевдокоде (на самом деле PHP), вы можете сами перевести его на java.
Алгоритм из здесь.

//$ipNetmask = "192.168.1.12/30";
list($ip, $netmask) = split( "/", $ipNetmask );
$ip_elements_decimal = split( "[.]", $ip );
$netmask_result="";
for($i=1; $i <= $netmask; $i++) {
  $netmask_result .= "1";
}
for($i=$netmask+1; $i <= 32; $i++) {
    $netmask_result .= "0";
}
$netmask_ip_binary_array = str_split( $netmask_result, 8 );
$netmask_ip_decimal_array = array();
foreach( $netmask_ip_binary_array as $k => $v ){
    $netmask_ip_decimal_array[$k] = bindec( $v ); // "100" => 4
    $network_address_array[$k] = ( $netmask_ip_decimal_array[$k] & $ip_elements_decimal[$k] );
}
$network_address = join( ".", $network_address_array );

// ------------------------------------------------
           // TCP/IP NETWORK INFORMATION
// ------------------------------------------------
// IP Entered = ..................: 192.168.1.12
// CIDR = ........................: /30
// Netmask = .....................: 255.255.255.252
// Network Address = .............: 192.168.1.12

// Broadcast Address = ...........: 192.168.1.15
// Usable IP Addresses = .........: 2
// First Usable IP Address = .....: 192.168.1.13
// Last Usable IP Address = ......: 192.168.1.14
person clyfe    schedule 31.05.2010

Вы можете использовать org.springframework.security.web.util.IpAddressMatcher из Spring Framework.

person Jiří Vypědřík    schedule 18.01.2012
comment
github.com/edazdarevic/CIDRUtils предоставляет вам поддержку IPv6 и тест, чтобы увидеть, находится ли IP-адрес в диапазон или нет. Однако длина маски отсутствует. См. Этот ответ stackoverflow.com/questions/558038/ - person Jan H; 10.06.2013

это мой заводной :)

//IP calculator by ku1gun

// input
String inputAddr = "12.34.56.78/20";

//magic
def(String ipAddrBin, String maskAddrBin, String invertedMaskBin, int hostsCount) = getIpAddrAndCidrMaskBin(inputAddr);

String broadcastAddr = retrieveBroadcastAddr(ipAddrBin, invertedMaskBin);
String ipAddr = getTenBaseAddrValueFromBin(ipAddrBin);
String maskAddr = getTenBaseAddrValueFromBin(maskAddrBin);
String invertedMask = getTenBaseAddrValueFromBin(invertedMaskBin);
String networkAddr = retrieveNetworkAddr(ipAddrBin, maskAddrBin);

def (String ipMinVal, String ipMaxVal) = getMinMaxIpRangeValues(networkAddr, broadcastAddr)

//Output "debug" results
System.out.println("Variables:");
System.out.println("ipInput: " + ipAddr);
System.out.println("MaskInput: " + maskAddr);
System.out.println("invertedMask: " + invertedMask);
System.out.println("-----------------------");
System.out.println("Binaries:");
System.out.println("ipBin: " + ipAddrBin);
System.out.println("MaskInBin: " + maskAddrBin);
System.out.println("InvertedMaskBin: " + invertedMaskBin);
System.out.println("-----------------------");
System.out.println("Results:");
System.out.println("maskAddr: " + maskAddr);
System.out.println("hostsCount: " + hostsCount);
System.out.println("networkAddr: " + networkAddr);
System.out.println("broadcastAddr: " + broadcastAddr);
System.out.println("ipMinVal: " + ipMinVal);
System.out.println("ipMaxVal: " + ipMaxVal);
System.out.println("-----------------------");
System.out.println("IP range list:");

long ipStart = host2long(ipMinVal);
long ipEnd = host2long(ipMaxVal);


for (long i=ipStart; i<=ipEnd; i++) 
{
    System.out.println(long2dotted(i));
}


//general methods
def getIpAddrAndCidrMaskBin(String inputAddrStr)
{
    def netMask = "";
    def invNetMask = "";

    def cidrAddressList = inputAddrStr.tokenize("\\/")
    def baseIPAddress = cidrAddressList.first()
    def cidrIPMask = cidrAddressList.last().toInteger()

    //retrieve binaryNetMask and binaryInvertedNetMask
    for(i=0; i<32; i++)
    {
        if(i<cidrIPMask)
        {
            netMask = netMask + "1";
            invNetMask = invNetMask + "0";
        }
        else
        {
            netMask = netMask + "0";
            invNetMask = invNetMask + "1";
        }
    }

    //retrieve binaryIpAddress
    String[] addrOctetArray = baseIPAddress.split("\\.");
    String binAddr = "";
    for (String string : addrOctetArray)
        {
            int octet = Integer.parseInt(string);     
            String binaryOctet = String.format("%8s", Integer.toBinaryString(octet)).replace(' ', '0');
            binAddr = binAddr + binaryOctet;
        }

    hostsCount = 2**(32 - cidrIPMask) - 2;

    return [binAddr, netMask, invNetMask, hostsCount]
}

def getTenBaseAddrValueFromBin(String binVal)
{
    tenBaseAddr = "";
    tenBaseAddr = tenBaseAddr + Integer.parseInt(binVal.substring(0,8), 2) + "." + Integer.parseInt(binVal.substring(8,16), 2) + "." + Integer.parseInt(binVal.substring(16,24), 2) + "." + Integer.parseInt(binVal.substring(24,32), 2)
    return tenBaseAddr;
}

def retrieveBroadcastAddr(String ipAddrBin, String invertedMaskBin)
{
    def oct_1 = Integer.parseInt(ipAddrBin.substring(0,8), 2) | Integer.parseInt(invertedMaskBin.substring(0,8), 2);
    def oct_2 = Integer.parseInt(ipAddrBin.substring(8,16), 2) | Integer.parseInt(invertedMaskBin.substring(8,16), 2);
    def oct_3 = Integer.parseInt(ipAddrBin.substring(16,24), 2) | Integer.parseInt(invertedMaskBin.substring(16,24), 2);
    def oct_4 = Integer.parseInt(ipAddrBin.substring(24,32), 2) | Integer.parseInt(invertedMaskBin.substring(24,32), 2);

    def t_oct = oct_1 + "."+ oct_2 + "." +  oct_3 + "." + oct_4;
    return t_oct
}

def retrieveNetworkAddr(String ipAddrBin, String maskInBin)
{
    def oct_1 = Integer.parseInt(ipAddrBin.substring(0,8), 2) & Integer.parseInt(maskInBin.substring(0,8), 2);
    def oct_2 = Integer.parseInt(ipAddrBin.substring(8,16), 2) & Integer.parseInt(maskInBin.substring(8,16), 2);
    def oct_3 = Integer.parseInt(ipAddrBin.substring(16,24), 2) & Integer.parseInt(maskInBin.substring(16,24), 2);
    def oct_4 = Integer.parseInt(ipAddrBin.substring(24,32), 2) & Integer.parseInt(maskInBin.substring(24,32), 2);

    def t_oct = oct_1 + "."+ oct_2 + "." +  oct_3 + "." + oct_4;
    return t_oct
}

def getMinMaxIpRangeValues(networkAddr, broadcastAddr)
{
    String[] ipAddrOctetArray = networkAddr.split("\\.");
    String[] broadcastOctetArray = broadcastAddr.split("\\.");

    String minRangeVal = ipAddrOctetArray[0] + "." + ipAddrOctetArray[1] + "." + ipAddrOctetArray[2] + "." + (Integer.parseInt(ipAddrOctetArray[3]) + 1)
    String maxRangeVal = broadcastOctetArray[0] + "." +broadcastOctetArray[1] + "." +broadcastOctetArray[2] + "." + (Integer.parseInt(broadcastOctetArray[3]) - 1)

    return[minRangeVal, maxRangeVal]
}

//IP list generate
public static long host2long(String host) 
{
    long ip=0;
    if (!Character.isDigit(host.charAt(0))) return -1;
    int[] addr = ip2intarray(host);
    if (addr == null) return -1;
    for (int i=0;i<addr.length;++i) 
    {
        ip += ((long)(addr[i]>=0 ? addr[i] : 0)) << 8*(3-i);
    }
    return ip;
}

public static int[] ip2intarray(String host) 
{
    Integer[] address = [-1,-1,-1,-1];
    int i=0;
    StringTokenizer tokens = new StringTokenizer(host,".");
    if (tokens.countTokens() > 4) return null;
    while (tokens.hasMoreTokens()) 
    {
        try 
        {
            address[i++] = Integer.parseInt(tokens.nextToken()) & 0xFF;
        } 
        catch(NumberFormatException nfe) 
        {
            return null;
        }
    }
    return address;
}

public static String long2dotted(long ip) 
{
    // if ip is bigger than 255.255.255.255 or smaller than 0.0.0.0
    if (ip > 4294967295l || ip < 0) 
    {
        throw new IllegalArgumentException("invalid ip");
    }
    StringBuilder ipAddress = new StringBuilder();
    for (int i = 3; i >= 0; i--) {
        int shift = i * 8;
        ipAddress.append((ip & (0xff << shift)) >> shift);
        if (i > 0) {
            ipAddress.append(".");
        }
    }
    return ipAddress.toString();
}
person ku1gun    schedule 29.02.2016
comment
это в основном уродливый Java-код с парой дефисов. В собственном Groovy это было бы вдвое меньше. - person Renato; 08.07.2016

Java-класс Apache SubnetUtils предлагает помощь в решении некоторых из этих задач:

String[] parts = ipv4Cidr.split("/");
if (parts[1].equals("0"))
{
    // This accepts all ip addresses.  Technically not a subnet.
    maskLength = 0;
    maskAdress = "0.0.0.0"
}
else
{
     maskLength = Integer.parseInt(parts[1]);
     cidrInfo = new SubnetUtils(ipv4Cidr).getInfo();
     maskAdress = cidrInfo.asInteger(cidrInfo.getNetmask());
     networkAddress = cidrInfo.getNetworkAddress()
}
person Adriana    schedule 11.05.2015

Вот простой пример Groovy

def cidrAddress = '192.168.10.0/24'
def cidrAddressList = cidrAddress.tokenize("\\/")
def baseIPAddress = cidrAddressList.first()
def cidrIPMask = cidrAddressList.last().toInteger()
def netMaskList = []
Integer fullOctets = cidrIPMask.intdiv(8)
fullOctets.times {netMaskList.add('255')}
def remainder = cidrIPMask % 8
netMaskList.add((256 - (2 ** (8 - remainder))).toString())
netMaskList.addAll(['0','0','0','0'])
def netMask = netMaskList.flatten().getAt(0..3).join('.')
return [cidrAddress,baseIPAddress,cidrIPMask,netMask]
person J Main    schedule 13.01.2016

person    schedule
comment
Добро пожаловать. Вы можете улучшить этот ответ, объяснив, как он решает проблему, поставленную в вопросе. - person Mark Chorley; 19.04.2016