需求
在某应用中,需要根据一定的规则生成随机的IP地址,规则类似于192.168.11.0/24这样的CIDR形式给出。
实现
经过艰苦卓绝的调试,下面的代码是可以用的:
1
2
3
4
5
6
7
8
9
10
11
12
|
RANDOM_IP_POOL = [ '192.168.10.222/0' ] def __get_random_ip(): str_ip = RANDOM_IP_POOL[random.randint( 0 , len (RANDOM_IP_POOL) - 1 )] str_ip_addr = str_ip.split( '/' )[ 0 ] str_ip_mask = str_ip.split( '/' )[ 1 ] ip_addr = struct.unpack( '>I' ,socket.inet_aton(str_ip_addr))[ 0 ] mask = 0x0 for i in range ( 31 , 31 - int (str_ip_mask), - 1 ): mask = mask | ( 1 << i) ip_addr_min = ip_addr & (mask & 0xffffffff ) ip_addr_max = ip_addr | (~mask & 0xffffffff ) return socket.inet_ntoa(struct.pack( '>I' , random.randint(ip_addr_min, ip_addr_max))) |
生成的结果如:
1
2
3
4
5
6
7
8
|
192.168 . 10.175 192.168 . 10.29 192.168 . 10.30 192.168 . 10.207 192.168 . 10.248 192.168 . 10.145 192.168 . 10.168 192.168 . 10.223 192.168 . 10.66 192.168 . 10.138 192.168 . 10.99 192.168 . 10.136 192.168 . 10.147 192.168 . 10.244 192.168 . 10.73 192.168 . 10.180 |
备注
(mask & 0xffffffff)这一用法,目的在于消除mask上32位以上的高位数据。由于mask是IPv4的掩码,所以不需要32位以上的数据。这一问题在mask取反之后比较明显。由于mask是一个32位以上的数(有可能是64位),取反之后,高位全部为1,计算的结果就不正确了。因此,在使用之前,需要利用(mask & 0xffffffff)清除高位。
供各位读者参考。