在使用 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/perl
use strict;
use warnings;
use Socket;
require 'sys/ioctl.ph';
sub get_ip_address($) {
my $pack = pack("a*", shift);
my $socket;
新聞熱點
疑難解答