大家討論使用Gearman做分散式處理時,各機需要註冊一個獨立的job作為資訊反饋,但是為了方便,Gearman::Worker指令碼register_function代碼又要通用,於是想到了使用各自的ip地址作為job命名~
那麼怎麼在worker指令碼裡擷取本機ip作為func呢?
第一種辦法,最簡單的,調用shell:
$ip = `ifconfig eth0|grep -oE '([0-9]{1,3}\.?){4}'|head -n 1`;
註:這裡輸入是固定的,所以簡單的[0-9]{1,3}了,如果是在web程式等地方驗證ip,需要更嚴謹!
或者
$ip = `ifconfig eth0|awk -F: '/inet addr/{split($2,a," ");print a[1];exit}'`;
好吧,這樣顯得太不perl了,而且頻繁的調用外部shell不太好,第二種:
open FH,"ifconfig eth0|";while(<FH>){last unless /inet addr:((\d{1,3}\.?){4})/;print $1;}
看起來稍微perl了一些,雖然實質跟上面的調用shell和grep法是一樣的。
第三種,更perl一點,純粹讀檔案:
open FH,'<','/etc/sysconfig/network-scripts/ifcfg-eth0';while(<FH>){next unless /IPADDR\s*=\s*(\S+)/;print $1;}
進一步的,如果不一定rh系,還要去讀/etc/issue,確定網路設定檔到底是/etc/sysconfig/network-script/ifcfg-eth0還是/etc/network/interfaces還是其他,然後根據不同發行版寫不同的處理方法……額,這是打算自己寫模組嗎?
好吧,大家來充分體會CPAN的魅力,去search一下,找到一把Sys::HostIP、Sys::HostAddr、Net::Inetface等模組。第四種:
use Sys::HostAddr;my $interface = Sys::HostAddr->new(ipv => '4', interface => 'eth0');print $interface->main_ip;
不過進去看看pm檔案,汗,這幾個模組都是調用ifconfig命令,不過是根據發行版的不同進行封裝而已。
還有辦法嗎?還有,看第五種:
perl -MPOSIX -MSocket -e 'my $host = (uname)[1];print inet_ntoa(scalar gethostbyname($host))';
不過有童鞋說了,這個可能因為hostname的原因,導致擷取的都是127.0.0.1……
那麼最後還有一招。通過strace ifconfig命令可以看到,linux實質是通過ioctl命令完成的網路介面ip擷取。那麼,我們也用ioctl就是了!
如下:
#!/usr/bin/perluse strict;use warnings;use Socket;require 'sys/ioctl.ph';sub get_ip_address($) {    my $pack = pack("a*", shift);    my $socket;    socket($socket, AF_INET, SOCK_DGRAM, 0);    ioctl($socket, SIOCGIFADDR(), $pack);    return inet_ntoa(substr($pack,20,4));};print get_ip_address("eth0");
這樣的好處,就是只調用了核心模組,在分髮腳本時,不用連帶安裝其他模組。
註:這個其實是根據網上有的一個py的指令碼修改的,py版如下:
#!/usr/bin/pythonimport socketimport fcntlimport structdef 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 get_ip_address('eth0')
原文:http://chenlinux.com/2011/09/20/get-ip-address-by-perl-on-linux/