如何ping一个IP地址
Posted
技术标签:
【中文标题】如何ping一个IP地址【英文标题】:How to ping an IP address 【发布时间】:2012-07-15 10:04:05 【问题描述】:我正在使用这部分代码在 java 中 ping 一个 IP 地址,但只有 ping localhost 是成功的,对于其他主机,程序说主机无法访问。 我禁用了我的防火墙,但仍然有这个问题
public static void main(String[] args) throws UnknownHostException, IOException
String ipAddress = "127.0.0.1";
InetAddress inet = InetAddress.getByName(ipAddress);
System.out.println("Sending Ping Request to " + ipAddress);
System.out.println(inet.isReachable(5000) ? "Host is reachable" : "Host is NOT reachable");
ipAddress = "173.194.32.38";
inet = InetAddress.getByName(ipAddress);
System.out.println("Sending Ping Request to " + ipAddress);
System.out.println(inet.isReachable(5000) ? "Host is reachable" : "Host is NOT reachable");
输出是:
向 127.0.0.1 发送 Ping 请求 主机可达 向 173.194.32.38 发送 Ping 请求 主机无法访问
【问题讨论】:
如果您直接使用 ping,是否可以 ping 该服务器? 你在 jTextField1 中输入了什么? 是的!但在我的程序中,我只能 ping localhost !!! @user1529128 我已经用一个简单的示例编辑了您的问题,该示例重现了您描述的行为。如果您认为这不是您所要求的,请随时回滚编辑。 @assylias:感谢您的精彩编辑!我是 *** 的新手,这是我的第一个问题。谢谢你帮助我。 【参考方案1】:InetAddress.isReachable()
根据javadoc:
".. 一个典型的实现将使用 ICMP ECHO REQUEST,如果 可以获取权限,否则会尝试建立TCP 目标主机的端口 7 (Echo) 上的连接..”。
选项 #1 (ICMP) 通常需要管理 (root)
权限。
【讨论】:
那么为什么您通常可以在大多数计算机上以非管理员/非 root 身份执行此操作Java 之外? 澄清一下:命令行 ping 等命令不使用 ICMP 数据包吗?我相当肯定他们做到了。 ping 是否在管理环境中运行? 你是说 isReachable 方法只使用 TCP 端口 7 而不会使用 ICMP ECHO 请求?或者如果我以某种方式向我的应用程序授予root权限,它会使用ICMP吗?如果是后者,那么如何让我的应用以 root 身份运行? Linux 上的ping
命令是set-uid root;这就是为什么非 root 用户可以使用它,即使它使用 ICMP ECHO_REQUEST。
这不太好用:***.com/questions/9922543/…【参考方案2】:
我认为这段代码会对你有所帮助:
public class PingExample
public static void main(String[] args)
try
InetAddress address = InetAddress.getByName("192.168.1.103");
boolean reachable = address.isReachable(10000);
System.out.println("Is host reachable? " + reachable);
catch (Exception e)
e.printStackTrace();
【讨论】:
【参考方案3】:检查您的连接。在我的电脑上,这两个 IP 都显示为 REACHABLE:
向 127.0.0.1 发送 Ping 请求 主机可达 向 173.194.32.38 发送 Ping 请求 主机可达
编辑:
您可以尝试修改代码以使用 getByAddress() 获取地址:
public static void main(String[] args) throws UnknownHostException, IOException
InetAddress inet;
inet = InetAddress.getByAddress(new byte[] 127, 0, 0, 1 );
System.out.println("Sending Ping Request to " + inet);
System.out.println(inet.isReachable(5000) ? "Host is reachable" : "Host is NOT reachable");
inet = InetAddress.getByAddress(new byte[] (byte) 173, (byte) 194, 32, 38 );
System.out.println("Sending Ping Request to " + inet);
System.out.println(inet.isReachable(5000) ? "Host is reachable" : "Host is NOT reachable");
getByName() 方法可能会尝试某种在您的机器上可能无法实现的反向 DNS 查找,getByAddress() 可能会绕过它。
【讨论】:
我确定我已连接,我什至禁用了我的防火墙和防病毒软件,但它仍然无法正常工作...... 它可能仍然是你的机器。你在什么操作系统上?您是否尝试过使用 getByAddress() 而不是 getByName() 方法来确保不尝试反向 DNS 查找? @Durandal 我正在使用win7,是的!我尝试了 getByAddress() 而不是 getByName() ,没有尝试 DNS 查找,请不要告诉我没有办法! @Durandal:为什么我们需要在第二个示例中将 ipaddress 类型转换为字节,你已经给出了? @vamsi 因为 IP 是作为文字给出的,并且编译器认为文字是 int 类型的。因为 byte 的值范围是 -128 到 127,超出该范围的值需要显式转换(例如文字 173)。 “为什么”源于 javas 字节被签名,而 IP 地址通常以无符号字节表示,以便于阅读。【参考方案4】:你不能简单地在 Java 中 ping,因为它依赖于 ICMP,遗憾的是 Java 不支持它
http://mindprod.com/jgloss/ping.html
改用套接字
希望对你有帮助
【讨论】:
我现在可能断章取义,但这对我有用,事实上非常好:dnsjava.org 谢谢大家。 下面有更好的答案 @AhmadArslan 无论您曾经在该链接上发布过什么答案,现在都没有了。所以这个链接不再有用了......你似乎已经杀死了你自己的链接......【参考方案5】:肯定会有用的
import java.io.*;
import java.util.*;
public class JavaPingExampleProgram
public static void main(String args[])
throws IOException
// create the ping command as a list of strings
JavaPingExampleProgram ping = new JavaPingExampleProgram();
List<String> commands = new ArrayList<String>();
commands.add("ping");
commands.add("-c");
commands.add("5");
commands.add("74.125.236.73");
ping.doCommand(commands);
public void doCommand(List<String> command)
throws IOException
String s = null;
ProcessBuilder pb = new ProcessBuilder(command);
Process process = pb.start();
BufferedReader stdInput = new BufferedReader(new InputStreamReader(process.getInputStream()));
BufferedReader stdError = new BufferedReader(new InputStreamReader(process.getErrorStream()));
// read the output from the command
System.out.println("Here is the standard output of the command:\n");
while ((s = stdInput.readLine()) != null)
System.out.println(s);
// read any errors from the attempted command
System.out.println("Here is the standard error of the command (if any):\n");
while ((s = stdError.readLine()) != null)
System.out.println(s);
【讨论】:
如果您将-c
更改为-n
,很有可能。
@Supuhstar 它仅适用于具有 ping 命令的操作系统。基本上它上面所做的就是分叉一个新进程并使其从操作系统运行ping
命令。【参考方案6】:
您可以使用此方法在 Windows 和其他平台上 ping 主机:
private static boolean ping(String host) throws IOException, InterruptedException
boolean isWindows = System.getProperty("os.name").toLowerCase().contains("win");
ProcessBuilder processBuilder = new ProcessBuilder("ping", isWindows? "-n" : "-c", "1", host);
Process proc = processBuilder.start();
int returnVal = proc.waitFor();
return returnVal == 0;
【讨论】:
在 windows 上你总是得到 1 作为 returnVal。不管它是否可达。 在 Windows 和 android 上都取得了成功 在没有 root 的情况下在 Windows 10 和 Android 7.1.1 上运行良好。【参考方案7】:简短推荐: 不要使用 isReachable(),调用系统 ping,如上面一些答案中所建议的那样。
长解释:
ping 使用 ICMP 网络协议。要使用 ICMP,需要一个“原始套接字” 操作系统不允许标准用户使用原始套接字 以下适用于fedora 30 linux,windows系统应该类似 如果 java 以 root 身份运行,isReachable() 实际上会发送 ICMP ping 请求 如果 java 没有以 root 身份运行,isReachable() 会尝试连接到 TCP 端口 7,即回显端口。该服务一般不再使用,尝试使用可能会产生不正确的结果 对连接请求的任何类型的回答,以及拒绝(TCP 标志 RST)都会从 isReachable() 中产生一个“真” 某些防火墙会为任何未明确打开的端口发送 RST。如果发生这种情况,您将获得 isReachable() == true 对于甚至不存在的主机 进一步尝试为 java 进程分配必要的功能: setcap cap_net_raw+eip java 可执行文件(分配使用原始套接字的权利) 测试:getcap java 可执行文件 -> 'cap_net_raw+eip'(已分配能力) 正在运行的 java 仍然向端口 7 发送 TCP 请求 使用 getpcaps pid 检查正在运行的 java 进程表明正在运行的 java 没有原始套接字功能。显然我的 setcap 已被某些安全机制覆盖 随着安全要求的增加,这可能会变得更加受限,除非 s.b.实现了一个特别针对 ping 的异常(但目前在网上找不到任何东西)【讨论】:
系统 ping 也不能保证正常工作;某些环境会在其防火墙处阻止所有 ICMP ECHO 请求。特别是,Azure(以及其他的 Github Actions)就是这样一种环境。【参考方案8】:
只是对其他人提供的内容的补充,即使它们运行良好,但在某些情况下,如果互联网速度较慢或存在一些未知的网络问题,某些代码将无法运行 (isReachable()
)。但是下面提到的这段代码创建了一个进程,该进程充当 Windows 的命令行 ping (cmd ping)。它在所有情况下都适用于我,经过尝试和测试。
代码:-
public class JavaPingApp
public static void runSystemCommand(String command)
try
Process p = Runtime.getRuntime().exec(command);
BufferedReader inputStream = new BufferedReader(
new InputStreamReader(p.getInputStream()));
String s = "";
// reading output stream of the command
while ((s = inputStream.readLine()) != null)
System.out.println(s);
catch (Exception e)
e.printStackTrace();
public static void main(String[] args)
String ip = "***.com"; //Any IP Address on your network / Web
runSystemCommand("ping " + ip);
希望对你有帮助,干杯!!!
【讨论】:
这可能容易受到命令注入的影响,请始终验证输入【参考方案9】:尽管这不依赖于 Windows 上的 ICMP,但此实现与新的 Duration API 配合得很好
public static Duration ping(String host)
Instant startTime = Instant.now();
try
InetAddress address = InetAddress.getByName(host);
if (address.isReachable(1000))
return Duration.between(startTime, Instant.now());
catch (IOException e)
// Host not available, nothing to do here
return Duration.ofDays(1);
【讨论】:
【参考方案10】:在带有 oracle-jdk 的 linux 上,OP 提交的代码在非 root 时使用端口 7,在 root 时使用 ICMP。当按照文档说明以 root 身份运行时,它确实会执行真正的 ICMP 回显请求。
如果您在 MS 机器上运行此应用程序,您可能必须以管理员身份运行应用程序才能获得 ICMP 行为。
【讨论】:
【参考方案11】:这是在Java
中ping IP 地址的方法,它应该适用于Windows
和Unix
系统:
import org.apache.commons.lang3.SystemUtils;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;
public class CommandLine
/**
* @param ipAddress The internet protocol address to ping
* @return True if the address is responsive, false otherwise
*/
public static boolean isReachable(String ipAddress) throws IOException
List<String> command = buildCommand(ipAddress);
ProcessBuilder processBuilder = new ProcessBuilder(command);
Process process = processBuilder.start();
try (BufferedReader standardOutput = new BufferedReader(new InputStreamReader(process.getInputStream())))
String outputLine;
while ((outputLine = standardOutput.readLine()) != null)
// Picks up Windows and Unix unreachable hosts
if (outputLine.toLowerCase().contains("destination host unreachable"))
return false;
return true;
private static List<String> buildCommand(String ipAddress)
List<String> command = new ArrayList<>();
command.add("ping");
if (SystemUtils.IS_OS_WINDOWS)
command.add("-n");
else if (SystemUtils.IS_OS_UNIX)
command.add("-c");
else
throw new UnsupportedOperationException("Unsupported operating system");
command.add("1");
command.add(ipAddress);
return command;
确保将Apache Commons Lang
添加到您的依赖项中。
【讨论】:
此示例可能不适用于非英语操作系统。在windows上,还有“一般故障”错误信息,可能还有其他的【参考方案12】:我更喜欢这种方式:
/**
*
* @param host
* @return true means ping success,false means ping fail.
* @throws IOException
* @throws InterruptedException
*/
private static boolean ping(String host) throws IOException, InterruptedException
boolean isWindows = System.getProperty("os.name").toLowerCase().contains("win");
ProcessBuilder processBuilder = new ProcessBuilder("ping", isWindows? "-n" : "-c", "1", host);
Process proc = processBuilder.start();
return proc.waitFor(200, TimeUnit.MILLISECONDS);
这种方式可以将阻塞时间限制在特定的时间,比如200毫秒。
适用于MacOS、Android和Windows,应该在JDK 1.8中使用。
这个想法来自Mohammad Banisaeid,但我无法评论。 (您必须有 50 声望才能发表评论)
【讨论】:
【参考方案13】:我知道以前的条目已经回答了这个问题,但是对于遇到这个问题的其他人,我确实找到了一种不需要在 Windows 中使用“ping”过程然后清理输出的方法。
我所做的是使用 JNA 调用 Window 的 IP 帮助程序库来执行 ICMP 回显
见my own answer to my own similar issue
【讨论】:
【参考方案14】:InetAddress 并不总是返回正确的值。在本地主机的情况下成功,但对于其他主机,这表明主机无法访问。尝试使用下面给出的 ping 命令。
try
String cmd = "cmd /C ping -n 1 " + ip + " | find \"TTL\"";
Process myProcess = Runtime.getRuntime().exec(cmd);
myProcess.waitFor();
if(myProcess.exitValue() == 0)
return true;
else
return false;
catch (Exception e)
e.printStackTrace();
return false;
【讨论】:
【参考方案15】:我尝试了几个选项:
-
Java InetAddress
InetAddress.getByName(ipAddress)
,Windows 上的网络在尝试了几次后开始出现异常
Java HttpURLConnection
URL siteURL = new URL(url);
connection = (HttpURLConnection) siteURL.openConnection();
connection.setRequestMethod("GET");
connection.setConnectTimeout(pingTime);
connection.connect();
code = connection.getResponseCode();
if (code == 200)
code = 200;
.
这很可靠,但有点慢
-
Windows 批处理文件
我终于决定在我的 Windows 机器上创建一个批处理文件,内容如下:ping.exe -n %echoCount% %pingIp%
然后我使用
public int pingBat(Network network)
ProcessBuilder pb = new ProcessBuilder(pingBatLocation);
Map<String, String> env = pb.environment();
env.put(
"echoCount", noOfPings + "");
env.put(
"pingIp", pingIp);
File outputFile = new File(outputFileLocation);
File errorFile = new File(errorFileLocation);
pb.redirectOutput(outputFile);
pb.redirectError(errorFile);
Process process;
try
process = pb.start();
process.waitFor();
String finalOutput = printFile(outputFile);
if (finalOutput != null && finalOutput.toLowerCase().contains("reply from"))
return 200;
else
return 202;
catch (IOException e)
log.debug(e.getMessage());
return 203;
catch (InterruptedException e)
log.debug(e.getMessage());
return 204;
事实证明这是最快最可靠的方法
【讨论】:
批处理示例不正确。输出很容易是“来自 x.y.z.a 的回复:无法访问目标主机”,这将返回一个肯定的答案。【参考方案16】:这应该可行:
import java.io.BufferedReader;
import java.io.InputStreamReader;
public class Pinger
private static String keyWordTolookFor = "average";
public Pinger()
// TODO Auto-generated constructor stub
public static void main(String[] args)
//Test the ping method on Windows.
System.out.println(ping("192.168.0.1"));
public String ping(String IP)
try
String line;
Process p = Runtime.getRuntime().exec("ping -n 1 " + IP);
BufferedReader input = new BufferedReader(new InputStreamReader(p.getInputStream()));
while (((line = input.readLine()) != null))
if (line.toLowerCase().indexOf(keyWordTolookFor.toLowerCase()) != -1)
String delims = "[ ]+";
String[] tokens = line.split(delims);
return tokens[tokens.length - 1];
input.close();
catch (Exception err)
err.printStackTrace();
return "Offline";
【讨论】:
请添加您的代码如何工作的描述。这将有助于 OP 在应用代码时理解您的代码。 有这个在“低质量队列”中进行审查......但是因为它的日期是一个小时......也许很快就会通过解释得到改进。急于发布 2012 年问题的答案?嗯...我现在就跳过它。 感谢您的所有 cmets。首先,这是一个仅适用于 Windows 的类和方法。我需要作为一个项目的 ping 时间,所以我想出了这个,如果你在 Windows 上工作,这很脏但很快。从代码本身可以看出,这是一个超快的约 5 分钟开发的用于 ping IPv4 主机的代码。此类及其方法获取一个字符串作为 IP 地址,并返回在 windows CMD 中生成的 ping 的平均时间。 CMD 那些 ping 和此代码读取输出以到达给出一些数字统计信息的行。 它在给出最小值、最大值和平均值时“标记”输出并返回平均时间。你可以只用一次 ping 来获得平均时间,这有点疯狂,但这是 Windows,无论如何它都会这样做。 请在标记之前在 Windows 上自己尝试一下。这是一个大问题的简单/肮脏的解决方案。以上是关于如何ping一个IP地址的主要内容,如果未能解决你的问题,请参考以下文章