4

假设我有一个文本文件包含一堆这样的 ip 范围:

x.x.x.x-y.y.y.y
x.x.x.x-y.y.y.y
x.x.x.x-y.y.y.y
x.x.x.x-y.y.y.y
x.x.x.x-y.y.y.y

xxxx 是起始值,yyyy 是范围的结束值。

如何在 python 的新文本文件中将这些 IP 范围转换为所有可能的 IP?

PS:这个问题和我之前的任何问题都不一样。我在上一个问题中问过“如何从 cidr 符号生成所有可能的 ips”。但是在这里我问“如何从ip范围列表生成”。这些是不同的东西。

4

4 回答 4

12

此函数从头到尾返回所有 IP 地址:

def ips(start, end):
    import socket, struct
    start = struct.unpack('>I', socket.inet_aton(start))[0]
    end = struct.unpack('>I', socket.inet_aton(end))[0]
    return [socket.inet_ntoa(struct.pack('>I', i)) for i in range(start, end)]

这些是您自己构建它的构建块:

>>> import socket, struct
>>> ip = '0.0.0.5'
>>> i = struct.unpack('>I', socket.inet_aton(ip))[0]
>>> i
5
>>> i += 1
>>> socket.inet_ntoa(struct.pack('>I', i))
'0.0.0.6'

例子:

ips('1.2.3.4', '1.2.4.5')
['1.2.3.4', '1.2.3.5', '1.2.3.6', '1.2.3.7', ..., '1.2.3.253', '1.2.3.254', '1.2.3.255', '1.2.4.0', '1.2.4.1', '1.2.4.2', '1.2.4.3', '1.2.4.4']

从文件中读取

在您的情况下,您可以从这样的文件中读取:

with open('file') as f:
    for line in f:
        start, end = line.strip().split('-')
        # ....
于 2013-07-14T17:02:21.470 回答
7

仅限 Python 3,对于 IPv4,与 @User 的想法相同,但使用新的 Python3 标准库:ipaddress

IPv4 由 4 个字节表示。所以next IP其实就是next number,一个IP范围可以表示为一个整数范围。

0.0.0.1 是 1

0.0.0.2 是 2

...

0.0.0.255 是 255

0.0.1.0 是 256

0.0.1.1 是 257

通过代码(忽略 In []: 和 Out []:)

In [68]: from ipaddress import ip_address

In [69]: ip_address('0.0.0.1')
Out[69]: IPv4Address('0.0.0.1')

In [70]: ip_address('0.0.0.1').packed
Out[70]: b'\x00\x00\x00\x01'

In [71]: int(ip_address('0.0.0.1').packed.hex(), 16)
Out[71]: 1

In [72]: int(ip_address('0.0.1.0').packed.hex(), 16)
Out[72]: 256

In [73]: int(ip_address('0.0.1.1').packed.hex(), 16)
Out[73]: 257

ip.packed.hex()返回 4 个字节的十六进制形式,因为它是十六进制,所以它更短(例如:0xff 十六进制 == 255 十进制 == 0b11111111 二进制),因此,通常用于表示字节。int(hex, 16)返回对应于十六进制值的整数值,因为它更人性化,并且可以用作 的输入ip_address

from ipaddress import ip_address

def ips(start, end):
    '''Return IPs in IPv4 range, inclusive.'''
    start_int = int(ip_address(start).packed.hex(), 16)
    end_int = int(ip_address(end).packed.hex(), 16)
    return [ip_address(ip).exploded for ip in range(start_int, end_int)]


ips('192.168.1.240', '192.168.2.5')

回报:

['192.168.1.240',
 '192.168.1.241',
 '192.168.1.242',
 '192.168.1.243',
 '192.168.1.244',
 '192.168.1.245',
 '192.168.1.246',
 '192.168.1.247',
 '192.168.1.248',
 '192.168.1.249',
 '192.168.1.250',
 '192.168.1.251',
 '192.168.1.252',
 '192.168.1.253',
 '192.168.1.254',
 '192.168.1.255',
 '192.168.2.0',
 '192.168.2.1',
 '192.168.2.2',
 '192.168.2.3',
 '192.168.2.4']
于 2017-10-18T02:20:08.530 回答
2

尝试这个:

def ip_range(start_ip,end_ip):
start = list(map(int,start_ip.split('.')))
end = list(map(int,end_ip.split('.')))
iprange=[]
while start!=list(map(int,end_ip.split('.'))):
    for i in range(len(start)-1,-1,-1):
        if start[i]<255:
            start[i]+=1
            break
        else:
            start[i]=0
    iprange.append('.'.join(map(str,start)))
return iprange
于 2014-09-17T00:42:49.753 回答
0

使用 ipaddress 和 range()

import ipaddress

print([ipaddress.ip_address(i).exploded for i in range(int(ipaddress.ip_address(first_ip)), int(ipaddress.ip_address(last_ip)))])
于 2022-02-16T07:58:16.887 回答