3

我正在构建一个嵌入式网络设备(基于 linux)并且遇到了动态构建守护进程 conf 文件的需求。因此,我需要能够在将构建 conf 文件的 python 代码中进行一些网络地址计算。我不是程序员,所以我担心我写了一个模块,一旦设备开始发货,它就无法像我希望的那样运行。

以下是我到目前为止所拥有的,它真的与我在这个网站和谷歌上可以找到的东西拼凑在一起。

有没有更好的方法来查找网络接口的网络地址和 cidr?将网络掩码转换为 bin str 并计算 1 似乎很不优雅。

import socket
import fcntl
import struct

SIOCGIFNETMASK = 0x891b
SIOCGIFADDR = 0x8915

s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

def _GetIfaceMask(iface):
    return struct.unpack('L', fcntl.ioctl(s, SIOCGIFNETMASK, struct.pack('256s', iface))[20:24])[0]

def _GetIfaceAddr(iface):
    return struct.unpack('L', fcntl.ioctl(s, SIOCGIFADDR, struct.pack('256s', iface[:15]))[20:24])[0]

def GetIfaceNet(iface):
    net_addr = _GetIfaceAddr(iface) & _GetIfaceMask(iface)
    return socket.inet_ntoa(struct.pack('L', net_addr))

def GetIfaceCidr(iface):
    bin_str = bin(_GetIfaceMask(iface))[2:]
    cidr = 0
    for c in bin_str:
        if c == '1':  cidr += 1
    return cidr

感谢您的任何意见,我真的有点迷失了。如果这不是此类反馈的地方,请告诉我。

4

2 回答 2

4

这可以使用汉明权重算法来解决。从如何计算 32 位整数中设置的位数?并翻译成 Python:

def number_of_set_bits(x):
    x -= (x >> 1) & 0x55555555
    x = ((x >> 2) & 0x33333333) + (x & 0x33333333)
    x = ((x >> 4) + x) & 0x0f0f0f0f
    x += x >> 8
    x += x >> 16
    return x & 0x0000003f

另一个更易读的解决方案(但在 中运行O(log x)):

def number_of_set_bits(x):
    n = 0
    while x:
        n += x & 1
        x = x >> 1
    return n
于 2011-02-06T10:25:27.857 回答
2

您可以查看 iptools python 模块http://code.google.com/p/python-iptools/它可以从长格式转换为点分格式,反之亦然。

于 2011-02-06T10:20:49.377 回答