7.4 列出网络上所有的活动主机
当我们涉及大型局域网时,可能需要检查网络上的其他主机是否处于活动状态。一台非活动主机可能有两种情况:要么是没有开机,要么是网络连接有问题。借助shell脚本,我们可以轻易找出并报告网络上的哪一台主机处于活动状态。让我们看看这是如何实现的。
7.4.1 新手上路
在这则攻略中,我们采用两种方法。第一种方法使用ping
,第二种方法使用fping
。fping
并没有默认包含在Linux发行版中。你可以用软件包管理器手动安装。
7.4.2 实战演练
让我们看看可以找出网络上所有活动主机的脚本,以及实现这一目标的其他方法。
- 方法1
我们可以用ping
命令编写脚本来查询一组IP地址同时检查它们是否处于活动状态:
#!/bin/bash
#文件名: ping.sh
# 用途:根据你的网络配置对网络地址192.168.0进行修改.
for ip in 192.168.0.{1..255} ;
do
ping $ip -c 2 &> /dev/null ;
if [ $? -eq 0 ];
then
echo $ip is alive
fi
done
输出如下:
- $ ./ping.sh
- 192.168.0.1 is alive
- 192.168.0.90 is alive
- 方法2
我们可以用已有的命令行工具来查询网络上的主机状态:
- $ fping -a 192.160.1/24 -g 2> /dev/null
- 192.168.0.1
- 192.168.0.90
或者,使用:
- $ fping -a 192.168.0.1 192.168.0.255 -g
7.4.3 工作原理
在方法1中,我们用ping
命令找出网络上的活动主机。我们用了一个for
循环对一组IP地址进行迭代。这组IP地址依照192.168.0.{1..255}格式生成。像{start..end}
这种记法会由shell对其进行扩展并生成一组IP地址,例如,192.168.0.1、192.168.0.2、192.168.0.3等依次类推,直到192.168.0.255。
ping $ip -c 2 &> /dev/null
会在每次循环中ping
对应的IP地址。-c 2
将发送的echo
分组数量限制为2。&> /dev/null
用于将stderr
和stdout
重定向到 /dev/null,使它们不会在终端上打印出来。我们用 $?
获取退出状态。如果顺利退出,退出状态为0,否则,为非0。因此能够ping
通的IP地址就被打印出来。我们也可以打印出不能ping
通的IP地址,从而给出一个无法抵达的IP地址列表。
这里为你准备了一个练习。不是在脚本中使用硬编码形式的IP地址范围,而是修改脚本使其能够从文件或
stdin
中读取一组IP地址。
在这个脚本中,每一次ping
都是依次执行的。即使所有的IP地址都是彼此独立,由于编写的是顺序式程序(sequential program),ping
命令也只能按顺序执行。每执行一次ping
命令,都要经历一段延迟:发送两个echo
分组,并接收回应或等待回应超时。
要是处理255个地址的话,这段延迟的时间可就不短了。我们可以使用并行方式来加速所有ping
命令的执行。这个脚本的核心部分是循环体。要使ping
命令可以并行执行,可将循环体放入( )&
。将命令块放入( )
,使其中的命令可作为子shell来执行,而&
可以使之脱离当前线程,在后台继续运行。例如:
- (
- ping $ip -c2 &> /dev/null ;
- if [ $? -eq 0 ];
- then
- echo $ip is alive
- fi
- )&
- wait
for
循环体执行多个后台进程,然后结束循环并终止脚本。要想等所有子进程结束之后再终止脚本,我们得使用wait
命令。将wait
放在脚本最后,它就会一直等到所有的子进程全部结束。
wait命令使脚本只有在所有的子进程或后台进程全部终止或完成之后才能结束。
请阅读本书所提供的脚本fast_ping.sh。
方法2使用了另一个命令fping
。它同时可以ping
一组IP地址,而且响应速度非常快。fping
的选项如下:
选项
-a
指定打印出所有活动主机的IP地址。选项
-u
指定打印出所有无法到达的主机。选项
-g
指定从写作IP/mask的“斜线-子网掩码”记法或者起止IP地址记法中生成IP地址范围。
- $ fping -a 192.160.1/24 -g
或者
- $ fping -a 192.160.1 192.168.0.255 -g
2>/dev/null
将由于主机无法到达所产生的错误信息打印到null设备。
也可以采用命令行参数的方式手动指定一组IP地址,或者从stdin
中接收。例如:
- $ fping -a 192.168.0.1 192.168.0.5 192.168.0.6
- # 将IP地址作为参数传递
- $ fping -a <ip.list
- # 从文件中传递一组IP地址
7.4.4 补充内容
fping
命令可以用来查询网络中的DNS数据。来看看它的具体用法。
用fping
进行DNS查找
fping
有一个选项-d
,它通过对每一个echo
回应进行DNS查询来返回主机名。使用该选项可以在ping的回应信息中打印出主机名而非IP地址。
- $ cat ip.list
- 192.168.0.86
- 192.168.0.9
- 192.168.0.6
- $ fping -a -d 2>/dev/null <ip.list
- www.local
- dnss.local
7.4.5 参考
1.5节讲解了数据重定向。
1.15节讲解了数字比较。