首选Java实现HTTP URL可用性的ping方式

173
我需要一个监视器类,定期检查给定的HTTP URL是否可用。我可以使用Spring TaskExecutor抽象来处理“定期”部分,所以这不是本文的重点。问题是:在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请求呢?
6个回答

291

这个会有用吗?(它能做我想要的吗?)

你可以这样做。另一种可行的方法是使用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。由于防火墙阻止其他端口,您可能会得到错误的负面结果。


我是否需要手动关闭连接?

不需要,连接已在幕后处理和汇集。


您可以将获取的URL连接转换为HttpURLConnection,然后使用setRequestMethod()设置请求方法。但是,需要注意的是,某些贫穷的Web应用程序或自制服务器可能会对HEAD返回HTTP 405错误(即不可用、未实现、不允许),而GET完全正常。如果您打算验证链接/资源而不是域名/主机,则使用GET更可靠。

在我的情况下,仅测试服务器的可用性是不够的,我需要测试URL(Web应用程序可能没有部署)

实际上,仅连接主机只能告知主机是否可用,而不能告知内容是否可用。有可能Web服务器已经启动但是Web应用程序在服务器启动期间未能成功部署。然而,这通常不会导致整个服务器崩溃。您可以通过检查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;
    }
}

4
谢谢提供详细信息,这些答案让SO成为一个很棒的地方。在我的情况下,仅测试服务器是否可用是不够的,我需要测试URL(Web应用程序可能未部署),所以我将坚持使用HttpURLConnection。关于HEAD方法不能作为很好的测试:如果我知道目标URL支持HEAD方法,那么它是一个好方法,我会检查一下。 - Sean Patrick Floyd
1
有些服务器可能会出现 java.io.IOException: unexpected end of stream 的问题,要解决它,您需要添加 connection.setRequestProperty("Accept-Encoding", "musixmatch")。这是一个已知的问题,并在 code.google.com 上报告了。 - Marcin Waśniowski
1
@BalusC因为(200 <= responseCode && responseCode <= 399)只有在(response <= 399)时才会是真的,这意味着(200 <= responseCode)条件是多余的。所以我认为这是一个错误。 - metator
5
@metator:哎???这绝对不是多余的。小于200的响应代码被认为是无效的。 - BalusC
1
@BalusC 在某些情况下,这些方法似乎不太有效。在这里看一下:https://dev59.com/Dl8e5IYBdhLWcg3wYJmu - AndreaF
显示剩余17条评论

18

使用URL对象调用openConnection()方法,而不是使用URLConnection。具体可以使用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();
        }
    }

还可以查看类似的问题:如何使用Java检查URL是否存在或返回404错误?

希望这可以帮助到你。


这适用于本地网络IP地址目标检查机制吗? - gumuruh

8

4
以下代码执行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;
    }
}

如果目标实际上是本地IP地址呢? - gumuruh

4
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;
}

可能的问题

  • 这真的足够快吗?是的,非常快!
  • 我不能只ping我自己的页面吗,毕竟我想请求它?当然可以!如果你想区分“互联网连接可用”和你自己的服务器是否可达,甚至可以同时检查两者。如果DNS出现问题怎么办?Google DNS(例如8.8.8.8)是世界上最大的公共DNS服务。截至2013年,它每天提供1300亿个请求。让我们说,你的应用程序不响应可能不会成为当天的话题。

阅读链接。它似乎很好

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

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

它们有一些不同,但在仅检查与互联网的连接功能方面,由于连接变量的原因,第一种方法可能会变得很慢。


2

考虑使用Restlet框架,该框架在这方面具有很好的语义。它强大而灵活。

代码可能非常简单:

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

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接