用于 ping HTTP URL 以获得可用性的首选 Java 方式

我需要一个定期检查给定 HTTP URL 是否可用的监视器类。我可以使用 SpringTaskExecator 抽象来处理“常规”部分,所以这不是这里的主题。问题是: 在 java 中 ping URL 的首选方法是什么?

以下是我当前的代码作为起点:

try {
final URLConnection connection = new URL(url).openConnection();
connection.connect();
LOG.info("Service " + url + " available, yeah!");
available = true;
} catch (final MalformedURLException e) {
throw new IllegalStateException("Bad URL: " + url, e);
} catch (final IOException e) {
LOG.info("Service " + url + " unavailable, oh no!", e);
available = false;
}
  1. 这一点都不好吗(它能做我想做的事情吗) ?
  2. 我是不是得想办法关闭连接?
  3. 我想这是一个 GET请求。有没有办法发送 HEAD代替?
201804 次浏览

您还可以使用 HttpURLConnection,它允许您设置请求方法(例如,设置为 HEAD)。显示如何发送请求、读取响应和断开连接的 举个例子

这到底有没有用(它能做我想做的事情吗?)

另一个可行的方法是使用 java.net.Socket

public static boolean pingHost(String host, int port, int timeout) {
try (Socket socket = new Socket()) {
socket.connect(new InetSocketAddress(host, port), timeout);
return true;
} catch (IOException e) {
return false; // Either timeout or unreachable or failed DNS lookup.
}
}

还有 InetAddress#isReachable():

boolean reachable = InetAddress.getByName(hostname).isReachable();

然而,这并没有显式地测试端口80。由于防火墙阻塞了其他端口,您可能会得到错误的否定。


我是不是得想办法关闭连接?

不,你不需要明确的。它的处理和汇集在引擎盖下。


我想这是一个 GET 请求。是否有一种方法来发送 HEAD 代替?

您可以将获得的 URLConnection强制转换为 HttpURLConnection,然后使用 setRequestMethod()设置请求方法。然而,您需要考虑到一些差劲的 Web 应用程序或本地服务器可能会返回一个 HEAD 的 HTTP 405错误(即不可用,不实现,不允许) ,而 GET 工作得非常好。如果您打算验证链接/资源而不是域/主机,那么使用 GET 更可靠。


在我的例子中,测试服务器的可用性是不够的,我需要测试 URL (webapp 可能不会被部署)

实际上,连接主机只通知主机是否可用,而不通知内容是否可用。一个 webserver 启动时没有出现问题,但是 webapp 在服务器启动时部署失败,这种情况也可能发生。然而,这通常不会导致整个服务器宕机。您可以通过检查 HTTP 响应代码是否为200来确定这一点。

HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();
connection.setRequestMethod("HEAD");
int responseCode = connection.getResponseCode();
if (responseCode != 200) {
// Not OK.
}


// < 100 is undetermined.
// 1nn is informal (shouldn't happen on a GET/HEAD)
// 2nn is success
// 3nn is redirect
// 4nn is client error
// 5nn is server error

有关响应状态代码的详细信息,请参阅 RFC 2616第10节。如果要确定响应数据,则不需要调用 connect()。它将隐式连接。

为了将来的参考,这里有一个实用方法的完整示例,也考虑了超时:

/**
* Pings a HTTP URL. This effectively sends a HEAD request and returns <code>true</code> if the response code is in
* the 200-399 range.
* @param url The HTTP URL to be pinged.
* @param timeout The timeout in millis for both the connection timeout and the response read timeout. Note that
* the total timeout is effectively two times the given timeout.
* @return <code>true</code> if the given HTTP URL has returned response code 200-399 on a HEAD request within the
* given timeout, otherwise <code>false</code>.
*/
public static boolean pingURL(String url, int timeout) {
url = url.replaceFirst("^https", "http"); // Otherwise an exception may be thrown on invalid SSL certificates.


try {
HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();
connection.setConnectTimeout(timeout);
connection.setReadTimeout(timeout);
connection.setRequestMethod("HEAD");
int responseCode = connection.getResponseCode();
return (200 <= responseCode && responseCode <= 399);
} catch (IOException exception) {
return false;
}
}

不要使用 URLConnection,而是通过在 URL 对象上调用 openConnection ()来使用 HttpURLConnection

然后使用 GetResponseCode ()将在您从连接中读取数据后提供 HTTP 响应。

这是密码:

    HttpURLConnection connection = null;
try {
URL u = new URL("http://www.google.com/");
connection = (HttpURLConnection) u.openConnection();
connection.setRequestMethod("HEAD");
int code = connection.getResponseCode();
System.out.println("" + code);
// You can determine on HTTP return code received. 200 is success.
} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
if (connection != null) {
connection.disconnect();
}
}

也检查类似的问题 如何检查 URL 是否存在或使用 Java 返回404?

希望这个能帮上忙。

考虑使用 Restlet 框架,它对这类事情有很好的语义。

代码可以很简单:

Client client = new Client(Protocol.HTTP);
Response response = client.get(url);
if (response.getStatus().isError()) {
// uh oh!
}

下面的代码执行 HEAD请求来检查网站是否可用。

public static boolean isReachable(String targetUrl) throws IOException
{
HttpURLConnection httpUrlConnection = (HttpURLConnection) new URL(
targetUrl).openConnection();
httpUrlConnection.setRequestMethod("HEAD");


try
{
int responseCode = httpUrlConnection.getResponseCode();


return responseCode == HttpURLConnection.HTTP_OK;
} catch (UnknownHostException noInternetConnection)
{
return false;
}
}
public boolean isOnline() {
Runtime runtime = Runtime.getRuntime();
try {
Process ipProcess = runtime.exec("/system/bin/ping -c 1 8.8.8.8");
int     exitValue = ipProcess.waitFor();
return (exitValue == 0);
} catch (IOException | InterruptedException e) { e.printStackTrace(); }
return false;
}

可能的问题

  • 这真的够快吗? 是的,非常快!
  • 我就不能发送我自己的主页吗,我想要的 当然! 如果你愿意,你甚至可以两个都查 区分「可使用互联网连接」与你自己的网络连接 服务器可以访问如果 DNS 关闭了怎么办。 8.8.8.8)是世界上最大的公共 DNS 服务。截至2013年,它每天处理1300亿个请求。这么说吧,你的应用程序没有 回应可能不会成为今天的话题。

看看这个链接,看起来很不错

编辑: 在我使用它的经验中,它没有这种方法快:

public boolean isOnline() {
NetworkInfo netInfo = connectivityManager.getActiveNetworkInfo();
return netInfo != null && netInfo.isConnectedOrConnecting();
}

他们有点不同,但在功能,只是检查连接到互联网的第一个方法可能会变慢,由于连接变量。