前言
在项目中有的时候需要检测两台服务器能不能正常通信,需要用到Ping命令。
目前已知的可分为两类方法:
- 使用Java API的InetAddress方式
- 使用Runtime.exec调用操作系统的命令CMD
使用InetAddress实现Ping
自Java 1.5开始,java.net包中就实现了ping的功能。详见InetAddress.isReachable()方法。
public static boolean ping(String ipAddress) throws Exception {
int timeOut = 3000 ; //超时应该在3钞以上
boolean status = InetAddress.getByName(ipAddress).isReachable(timeOut);
// 当返回值是true时,说明host是可用的,false则不可。
return status;
}
一般我们的应用只在内网ping的话可以使用这种方式,外网不推荐。
isReachable使用的是端口7来实现的ping命令而不是使用ICMP协议,因此在该方法在尝试连接有防火墙配置的远程服务器是会出现不可达。
调用CMD
通过程序调用类似“ping 127.0.0.1 -n 4 -w 3000”的命令,该命令ping10次,单次间隔3秒。
网络通的情况会输出:
C:\Users\tgg>ping 127.0.0.1 -n 10 -w 3000 正在 Ping 127.0.0.1 具有 32 字节的数据: 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 来自 127.0.0.1 的回复: 字节=32 时间<1ms TTL=64 127.0.0.1 的 Ping 统计信息: 数据包: 已发送 = 10,已接收 = 10,丢失 = 0 (0% 丢失), 往返行程的估计时间(以毫秒为单位): 最短 = 0ms,最长 = 0ms,平均 = 0ms
以上信息输出是根据操作系统的语言来进行本地化的,其中"ms TTL="
是不变的,我们可以通过Runtime.exec方法来调用本地CMD命令来执行以上语句,代码如下:
import org.apache.log4j.Logger;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.InetAddress;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/** * @author tgg */
public class Ping {
public static boolean ping(String ipAddress) throws Exception {
int timeOut = 3000 ;
boolean status = InetAddress.getByName(ipAddress).isReachable(timeOut);
return status;
}
public static boolean ping(String ipAddress, int pingTimes, int timeOut) {
BufferedReader in = null;
Runtime r = Runtime.getRuntime();
// 将要执行的ping命令,此命令是windows格式的命令
String pingCommand = "ping " + ipAddress + " -n " + pingTimes + " -w " + timeOut;
// Linux命令如下
// String pingCommand = "ping" -c " + pingTimes + " -w " + timeOut + ipAddress;
try {
if (logger.isDebugEnabled()) {
logger.debug(pingCommand);
}
// 执行命令并获取输出
Process p = r.exec(pingCommand);
if (p == null) {
return false;
}
in = new BufferedReader(new InputStreamReader(p.getInputStream()));
int connectedCount = 0;
String line;
// 逐行检查输出,计算类似出现=23ms TTL=62字样的次数
while ((line = in.readLine()) != null) {
connectedCount += getCheckResult(line);
}
// 如果出现类似=23ms TTL=62这样的字样,出现的次数=测试次数则返回真
return connectedCount == pingTimes;
} catch (Exception e) {
logger.error(e);
return false;
} finally {
try {
in.close();
} catch (IOException e) {
logger.error(e);
}
}
}
//若line含有=18ms TTL=16字样,说明已经ping通,返回1,否則返回0.
private static int getCheckResult(String line) { // System.out.println("控制台输出的结果为:"+line);
Pattern pattern = Pattern.compile("(\\d+ms)(\\s+)(TTL=\\d+)", Pattern.CASE_INSENSITIVE);
Matcher matcher = pattern.matcher(line);
while (matcher.find()) {
return 1;
}
return 0;
}
private static final Logger logger = Logger.getLogger(Ping.class);
}