如何从接口获取物理接口 IP 地址

Posted

技术标签:

【中文标题】如何从接口获取物理接口 IP 地址【英文标题】:How to get the physical interface IP address from an interface 【发布时间】:2011-09-08 18:05:15 【问题描述】:

到目前为止,我使用 PyQt 类做了什么:

all_Addresses = QNetworkInterface.allAddresses()    #list-of-QHostAddress

for addr in all_Addresses:
    print(addr.toString())

输出:

172.16.0.186 - Virtual Interface IP address
192.168.10.2 - Physical interface IP address. I want this one.
127.0.0.1

使用socket

import socket
print(socket.gethostbyname(socket.gethostname()))

输出:

172.16.0.186 - When open*** is on
192.168.10.2 - When its off
    有没有办法区分它们? 这可以用普通的 Python 而不是 PyQt 类来完成吗? 如何获得 IPv6 地址?

【问题讨论】:

Finding local IP addresses in Python.的可能重复 环回总是127.0.0.1,所以这几乎不是问题。 顺便说一下地址方案中127.0.0.1localhost相同与::-1相同 感谢您的快速答复。我尝试了 Lennart 提供的示例,使用导入套接字。存在一个接口可以有多个IP地址的问题,并且在使用***时使用套接字方式显示不同的输出。 你想达到什么目的? 【参考方案1】:

您应该使用netifaces。它被设计为跨平台的,包含适用于 Windows 的专用代码以及适用于不同 UNIX/类 UNIX 平台的各种通用版本。

从netifaces 0.10.0 版开始,支持 Python3。

使用总结

>>> from netifaces import AF_INET, AF_INET6, AF_LINK, AF_PACKET, AF_BRIDGE
>>> import netifaces as ni
>>> ni.interfaces()
['lo', 'eth0', 'eth1', 'vboxnet0', 'dummy1']
>>>
>>> ni.ifaddresses('eth0')[AF_LINK]   # NOTE: AF_LINK is an alias for AF_PACKET
['broadcast': 'ff:ff:ff:ff:ff:ff', 'addr': '00:02:55:7b:b2:f6']
>>> ni.ifaddresses('eth0')[AF_INET]
['broadcast': '172.16.161.7', 'netmask': '255.255.255.248', 'addr': '172.16.161.6']
>>>
>>> # eth0 ipv4 interface address
>>> ni.ifaddresses('eth0')[AF_INET][0]['addr']
'172.16.161.6'
>>>>

详情

Windows 支持:

大多数 MS Windows 安装不需要编译器。如果您收到有关安装 MS Visual C++ for Windows 的警告,请发送电子邮件至 very careful,因为您需要将用于 python 的编译器版本与用于模块的编译器版本相匹配。

netifaces数据结构的详细示例:

>>> import netifaces as ni
>>> ni.interfaces()
['lo', 'eth0', 'eth1', 'vboxnet0', 'dummy1']
>>> ni.ifaddresses('eth0')

    17: [
        
            'broadcast': 'ff:ff:ff:ff:ff:ff',
            'addr': '00:02:55:7b:b2:f6'
        
    ],
    2: [
        
            'broadcast': '172.16.161.7',
            'netmask': '255.255.255.248',
            'addr': '172.16.161.6'
        
    ],
    10: [
        
            'netmask': 'ffff:ffff:ffff:ffff::',
            'addr': 'fe80::202:55ff:fe7b:b2f6%eth0'
        
    ]

>>> 
>>> print(ni.ifaddresses.__doc__)
Obtain information about the specified network interface.

Returns a dict whose keys are equal to the address family constants,
e.g. netifaces.AF_INET, and whose values are a list of addresses in
that family that are attached to the network interface.
>>>
>>> # for the IPv4 address of eth0
>>> ni.ifaddresses('eth0')[2][0]['addr']
'172.16.161.6'

用于索引协议的数字来自/usr/include/linux/socket.h(在 Linux 中)...编辑:我的 3.2 内核在这里有它们:/usr/src/linux-headers-3.2.0-4-common/include/linux/socket.h

#define AF_INET         2       /* Internet IP Protocol         */
#define AF_INET6        10      /* IP version 6                 */
#define AF_PACKET       17      /* Packet family                */

好消息是您不必记住所有这些标头常量,它们包含在 netifaces 中:

>>> from netifaces import AF_INET, AF_INET6, AF_LINK, AF_PACKET, AF_BRIDGE
>>> import netifaces as ni

【讨论】:

它看起来很有趣,但我无法让它工作。我在安装easy_install 时遇到了麻烦,这看起来是netifaces 的要求。对不起,我是 Python 新手 :-( @Johny Mnemonic,你不需要easy_install...只需download the source,用tar xvfz netifaces-0.5.tar.gz解压并以root身份安装:cd netifaces-0.5/; python setup.py install netifaces 一切都很好,但它不支持 py3k,提供此类的克隆仍然需要 Windows 上的 C 编译器。 netifaces 0.10.0 及更高版本支持 Py3K 并且还有 Windows 二进制文件,因此您不需要编译器。 在 ubuntu 上 sudo apt-get install python-netifaces 或对于 python3 sudo apt-get install python3-netifaces【参考方案2】:

使用 Linux SIOCGIFADDR ioctl 查找与网络接口关联的 IP 地址,给定该接口的名称,例如“eth0”。地址以包含点分四边形的字符串形式返回。

import socket
import fcntl
import struct

def get_ip_address(ifname):
    s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
    return socket.inet_ntoa(fcntl.ioctl(
        s.fileno(),
        0x8915,  # SIOCGIFADDR
        struct.pack('256s', ifname[:15])
    )[20:24])

>>> get_ip_address('lo')
'127.0.0.1'

>>> get_ip_address('eth0')
'38.113.228.130'

For more

【讨论】:

但它不适用于 python3.4.3,低于错误,Traceback (most recent call last): File "/home/user1/ipadd.py", line 27, in <module> defaultIpAddr = get_ip_address(defaultIfaceName) File "/home/user1/ipadd.py", line 24, in get_ip_address struct.pack('256s', ifname[:15]) struct.error: argument for 's' must be a bytes object 在 python 3.4 上这个函数不起作用,像这样将 ifname 转换为字节数组 (ifname[:15] .encode('utf-8')) 当接口有多个地址时,我不理解神奇的 ioctl-address 摆弄来获取所有地址。你能详细说明一下代码吗?【参考方案3】:

我使用这个解决方案。其实有点棘手,而且只适用于linux家族。

import commands
intf = 'eth0'
intf_ip = commands.getoutput("ip address show dev " + intf).split()
intf_ip = intf_ip[intf_ip.index('inet') + 1].split('/')[0]
print intf_ip

这些代码在 linux 系列操作系统上使用ip 命令。它拆分ip 命令的输出,并且只获取这些接口的 IPv4 地址。您可以将值intf 更改为eth1p2p1

【讨论】:

以上是关于如何从接口获取物理接口 IP 地址的主要内容,如果未能解决你的问题,请参考以下文章

linuxvlan虚拟接口如何和物理接口通

请问:思科的三层交换机可以给每个物理接口配一个ip地址么?

如何从本地接口获取所有地址和掩码?

路由器接口的IP地址配错了该怎么处理

在 Linux 上获取接口的 IP 地址

如何使用 C 获取 OS X 上所有接口的 IP 地址