見過很多擷取伺服器本地IP的代碼,個人覺得都不是很好,例如以下這些
不推薦:靠猜測去擷取本地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,那麼此類代碼都有可能失敗,而且還不容易移植到其他平台上。
不推薦:通過hostname來擷取本機IP
import socket
print(socket.gethostbyname(socket.gethostname()))
# Python學習交流QQ群:857662006
# 有可能出現這個情況
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方法了。沒有任何的依賴,也沒有去猜測機器上的網絡裝置資訊。
而且是利用 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