当前位置: 首页 > 面试题库 >

ping HTTP URL以获取可用性的首选Java方法

沈俊晤
2023-03-14
问题内容

我需要一个监视器类,该类定期检查给定的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呢?

问题答案:

这有什么好处吗(会做我想要的吗?)

你可以这样做。另一种可行的方法是使用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()设置请求方法。但是,你需要考虑到,当GET正常运行时,某些性能较差的Web应用程序或本地服务器可能为HEAD 返回HTTP 405错误(即,不可用,未实现,不允许)。如果你打算验证链接/资源而不是域/主机,则使用GET更可靠。

在我的情况下,仅测试服务器的可用性是不够的,我需要测试URL(可能未部署webapp)

实际上,连接主机仅通知主机是否可用,而不通知内容是否可用。可能发生的情况是,网络服务器启动时没有问题,但是在服务器启动期间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;
    }
}


 类似资料:
  • 问题内容: 嗨, 这是从JComboBox作为字符串获取值的正确方法,以及为什么它是正确的方法。谢谢。 要么 问题答案: 如果仅在JComboBox中放置(非空)引用,则两种方法都可以。 但是,第一种解决方案还允许将来进行修改,您可以在组合框中插入s,s,s等作为项目。 为了对值保持鲁棒性(仍然不强制转换),您可以考虑第三个选择:

  • 本文向大家介绍Java获取首选项的子节点,包括了Java获取首选项的子节点的使用技巧和注意事项,需要的朋友参考一下 示例 Preferences对象始终代表整个Preferences树中的特定节点,如下所示: 要选择/com/mycompany/myapp节点: 按照惯例,基于一个类的包: 通过相对路径: 使用相对路径(不是以开头的路径/)将导致该路径相对于在其上解析的父节点进行解析。例如,以下示

  • 获取系统首选项 进程: 主进程 例子: 1 const {systemPreferences} = require('electron') 2 console.log(systemPreferences.isDarkMode()) Copied! 事件列表 systemPreferences 对象提供以下事件: 事件: 'accent-color-changed' Windows 触发:用户在个性

  • 问题内容: 在Java中,先实例化ZipOutputStream还是先实例化BufferedOutputStream 无关紧要?例: 要么: 在我不科学的时光里,我似乎并没有说出太多区别。我在Java API中看不到任何说明这些方法之一是必需的还是首选的信息。有什么建议吗?似乎先压缩输出,然后将其缓冲以进行写操作会更有效。 问题答案: 你应该总是包裹用,绝不是相反。请参见以下代码: 如评论所述,和

  • 问题内容: 我想知道用Java加载资源的最佳方法: this.getClass().getResource() (or getResourceAsStream()), Thread.currentThread().getContextClassLoader().getResource(name), System.class.getResource(name)。 问题答案: 根据你的需要制定解决方案

  • 问题内容: 如果我有这个: 如果可能的话,如何在不实际调用方法的情况下获得对f.bar的引用? 编辑添加:我想做的是编写一个对f的成员进行迭代并对其执行某些操作的函数(不重要)。属性让我感到震惊,因为仅在getattr()中命名它们会调用其__get __()方法。 问题答案: (下图)在给定对象的中查找,如果存在则返回关联的值。如果不是其中的关键,则搜索对象的MRO 。如果找不到密钥,则引发。