Python 優雅獲取本機 IP 方法

原文python

見過不少獲取服務器本地IP的代碼,我的以爲都不是很好,例如如下這些shell

不推薦:靠猜想去獲取本地IP方法

#!/usr/bin/env python
# -*- coding: utf-8 -*-
 
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])
 
print "br1 = "+ get_ip_address('br1')
print "lo = " + get_ip_address('lo')
print "virbr0 = " + get_ip_address('virbr0')

這類代碼帶有猜想的行爲。緩存

若是機器上只有eth0 或者 只有bond0上有IP,那麼此類代碼都有可能失敗,並且還不容易移植到其餘平臺上。bash

不推薦:經過hostname來獲取本機IP

import socket
print(socket.gethostbyname(socket.gethostname()))
 
# 有可能出現這個狀況
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
socket.gaierror: [Errno -2] Name or service not known

這個方法是經過獲取hostname,而後再經過hostname反查處機器的IP。這個方法也是不推薦的。由於不少的機器沒有規範這個hostname的設置。服務器

另外就是有些服務器會在 /etc/hosts 中添加本機的hostname的地址,這個作法也不是不能夠,可是若是設置成了 127.0.0.1,那麼獲取出來的IP就都是這個地址了。網絡

經過 UDP 獲取本機 IP,目前見過最優雅的方法

這個方法是目前見過最優雅獲取本機服務器的IP方法了。沒有任何的依賴,也沒有去猜想機器上的網絡設備信息。socket

並且是利用 UDP 協議來實現的,生成一個UDP包,把本身的 IP 放如到 UDP 協議頭中,而後從UDP包中獲取本機的IP。函數

這個方法並不會真實的向外部發包,因此用抓包工具是看不到的。可是會申請一個 UDP 的端口,因此若是常常調用也會比較耗時的,這裏若是須要能夠將查詢到的IP給緩存起來,性能能夠得到很大提高。工具

# 在 shell 中能夠一行調用,獲取到本機IP
python -c "import socket;print([(s.connect(('8.8.8.8', 53)), s.getsockname()[0], s.close()) for s in [socket.socket(socket.AF_INET, socket.SOCK_DGRAM)]][0][1])"
10.12.189.16
# 能夠封裝成函數,方便 Python 的程序調用
import socket
 
def get_host_ip():
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
        s.connect(('8.8.8.8', 80))
        ip = s.getsockname()[0]
    finally:
        s.close()
 
    return ip
相關文章
相關標籤/搜索