我可以在Android中为DefaultHttpClient设置getaddrinfo超时吗?

时间:2013-08-13 19:16:50

标签: java android http networking connectivity

在Android应用中,我正在尝试测试用户是否有可用的Internet连接。如果您有兴趣,可以在上一个问题Detecting limited network connectivity in Android?

中找到一些背景知识

代码基本上就像:

try {
    HttpParams myParams = new BasicHttpParams();
    HttpConnectionParams.setConnectionTimeout(myParams, 10000);
    HttpConnectionParams.setSoTimeout(myParams, 10000);

    httpClient = new DefaultHttpClient(myParams);

    request = new HttpHead(url);
    response = httpClient.execute(request);
    statusCode = response.getStatusLine().getStatusCode();

    if (statusCode != 200)
    {       
        return false;
    }
    return true;

} catch(Exception e) {
    return false;
}

我可以使用HttpConnectionParams控制Connection和Socket的超时。但是,如果我的设备连接到Wifi,但wifi无法访问Internet,我在异常中遇到的错误是:

  

libcore.io.GaiException:getaddrinfo failed:EAI_NODATA(无地址   与主机名相关联)

     

无法解析主机“www.example.com”:没有与之关联的地址   主机名

看起来它在DNS查找上超时了。我可以控制DNS查找的超时吗? httpClient.execute大约需要45秒才能失败,但上述情况除外。我希望它能早点放弃。

1 个答案:

答案 0 :(得分:9)

我做了一些功课,似乎你无法调整DNS查找超时。因此,我认为更好的方法是显式DNS查找,以便我可以控制它(并希望缓存结果以加快下一次尝试)。所以这让我变得简单:

InetAddress addr = InetAddress.getByName(hostname);

但这也有45秒超时。其他人提到有no control of the timeout for getByName()。最后,我偶然发现了一个简单的解决方案,只需在一个单独的线程中启动查找并管理自己的超时。 This blog post很好地证明了这一点。

private static boolean testDNS(String hostname) {
  try
  {
    DNSResolver dnsRes = new DNSResolver(hostname);
    Thread t = new Thread(dnsRes);
    t.start();
    t.join(1000);
    InetAddress inetAddr = dnsRes.get();            
    return inetAddr != null;
  }
  catch(Exception e)
  { 
    return false;
  }
}

private static class DNSResolver implements Runnable {
    private String domain;
    private InetAddress inetAddr;

    public DNSResolver(String domain) {
        this.domain = domain;
    }

    public void run() {
        try {
            InetAddress addr = InetAddress.getByName(domain);
            set(addr);
        } catch (UnknownHostException e) {                
        }
    }

    public synchronized void set(InetAddress inetAddr) {
        this.inetAddr = inetAddr;
    }
    public synchronized InetAddress get() {
        return inetAddr;
    }
}

使用此功能,我可以先测试设备是否可以解析主机名,然后测试是否成功进行完全连接测试。