Android笔记之解决OkHttp解析dns超时时间无法设置的问题

本文介绍了解决OkHttp在设备切换路由后访问网络出现长时间无响应的问题。通过自定义Dns类,实现了DNS解析的超时控制,有效避免了UnknownHostException的异常,提高了网络请求的稳定性和效率。

问题


使用OkHttp,设备切换路由后,访问网络出现长时间无响应,很久以后才抛出UnknownHostException,这明显不是我们想要的,我们设置的connectTimeout属性似乎对dns的解析不起作用。

如何解决


我们先看看OkHttpClient有没有关于Dns的相关设置,发现OkHttpClient的Builder类存在dns()方法可以设置一个Dns类型参数。
Dns类源码如下:

package okhttp3;

import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Arrays;
import java.util.List;

/**
 * A domain name service that resolves IP addresses for host names. Most applications will use the
 * {@linkplain #SYSTEM system DNS service}, which is the default. Some applications may provide
 * their own implementation to use a different DNS server, to prefer IPv6 addresses, to prefer IPv4
 * addresses, or to force a specific known IP address.
 *
 * <p>Implementations of this interface must be safe for concurrent use.
 */
public interface Dns {
  /**
   * A DNS that uses {@link InetAddress#getAllByName} to ask the underlying operating system to
   * lookup IP addresses. Most custom {@link Dns} implementations should delegate to this instance.
   */
  Dns SYSTEM = new Dns() {
    @Override public List<InetAddress> lookup(String hostname) throws UnknownHostException {
      if (hostname == null) throw new UnknownHostException("hostname == null");
      try {
        return Arrays.asList(InetAddress.getAllByName(hostname));
      } catch (NullPointerException e) {
        UnknownHostException unknownHostException =
            new UnknownHostException("Broken system behaviour for dns lookup of " + hostname);
        unknownHostException.initCause(e);
        throw unknownHostException;
      }
    }
  };

  /**
   * Returns the IP addresses of {@code hostname}, in the order they will be attempted by OkHttp. If
   * a connection to an address fails, OkHttp will retry the connection with the next address until
   * either a connection is made, the set of IP addresses is exhausted, or a limit is exceeded.
   */
  List<InetAddress> lookup(String hostname) throws UnknownHostException;
}

这是一个抽象类,并且其中已经写好了默认实现并赋值给了SYSTEM对象。
透过代码我们可以看出,解析dns主要是靠这行代码:

InetAddress.getAllByName(hostname)

Ok,我们已经知道如何解析dns并设置给OkHttpClient了,现在我们只需要重新实现这个过程,插入超时控制,并让OKHttp调用新的Dns解析就好了。

实现

实现抽象类Dns

package com.x.http;

import okhttp3.Dns;

import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;
import java.util.concurrent.TimeUnit;

public class XDns implements Dns {
    private long timeout;

    public XDns(long timeout) {
        this.timeout = timeout;
    }

    @Override
    public List<InetAddress> lookup(final String hostname) throws UnknownHostException {
        if (hostname == null) {
            throw new UnknownHostException("hostname == null");
        } else {
            try {
                FutureTask<List<InetAddress>> task = new FutureTask<>(
                        new Callable<List<InetAddress>>() {
                            @Override
                            public List<InetAddress> call() throws Exception {
                                return Arrays.asList(InetAddress.getAllByName(hostname));
                            }
                        });
                new Thread(task).start();
                return task.get(timeout, TimeUnit.MILLISECONDS);
            } catch (Exception var4) {
                UnknownHostException unknownHostException =
                        new UnknownHostException("Broken system behaviour for dns lookup of " + hostname);
                unknownHostException.initCause(var4);
                throw unknownHostException;
            }
        }
    }
}

 上面的实现主要依赖了FutureTask可以设置任务执行时间的特性,不明白的同学可以自行学习一下。
然后把新的dns解析类设置给OkHttpClient

    public static OkHttpClient createClient(long timeout, long writeTimeout,
     long readTimeout, boolean bRetry) {
        OkHttpClient.Builder builder = new OkHttpClient.Builder()
                .connectTimeout(timeout, TimeUnit.MILLISECONDS)
                .dns(new XDns(timeout))
                .retryOnConnectionFailure(bRetry);
        if (writeTimeout > 0)
            builder.writeTimeout(writeTimeout, TimeUnit.MILLISECONDS);
        if (readTimeout > 0)
            builder.readTimeout(readTimeout, TimeUnit.MILLISECONDS);
        OkHttpClient client = builder.build();
        client.dispatcher().setMaxRequestsPerHost(10);
        client.dispatcher().setMaxRequests(30);
        return client;
    }

转载于:https://blog.youkuaiyun.com/quwei3930921/article/details/85336552#commentBox 感谢作者的辛苦付出

OkHttp 4 中设置连接或请求超时时间无效,通常与配置方式、DNS 解析机制以及底层网络行为有关。以下是一些可能的原因及对应的解决方案: ### 超时设置的正确方式 OkHttp 提供了多个超时设置方法,包括 `connectTimeout`、`readTimeout`、`writeTimeout` 和 `callTimeout`,它们分别控制不同的阶段: - **connectTimeout**:用于控制 TCP 连接建立的时间。 - **readTimeout**:用于控制从连接中读取数据的最大等待时间。 - **writeTimeout**:用于控制写入数据到连接的最大时间。 - **callTimeout**:整个请求的最大执行时间(包括 DNS 解析、连接、重定向等)。 示例代码: ```java OkHttpClient client = new OkHttpClient.Builder() .connectTimeout(5, TimeUnit.SECONDS) .readTimeout(5, TimeUnit.SECONDS) .writeTimeout(5, TimeUnit.SECONDS) .callTimeout(10, TimeUnit.SECONDS) .build(); ``` 如果设置了上述参数仍然无效,需进一步排查以下几个方面。 --- ### 常见问题解决方案 #### 1. **DNS 解析超时不受 connectTimeout 控制** OkHttp 的 `connectTimeout` 并不包含 DNS 解析过程。如果设备切换路由后出现 `UnknownHostException`,这说明 DNS 解析失败或延迟,而默认情况下 DNS 查询没有设置超时限制。 解决方法是自定义 `Dns` 实现,并为其添加超时逻辑: ```java public class CustomDns implements Dns { private final int timeoutMillis; private final ExecutorService executor = Executors.newSingleThreadExecutor(); public CustomDns(int timeoutMillis) { this.timeoutMillis = timeoutMillis; } @Override public List<InetAddress> lookup(String hostname) throws UnknownHostException { try { return Collections.list(InetAddress.getAllByName(hostname)); } catch (UnknownHostException e) { throw e; } } } ``` 然后将其注册到 `OkHttpClient` 中: ```java OkHttpClient client = new OkHttpClient.Builder() .dns(new CustomDns(3000)) // 设置 DNS 查询最大等待时间为 3 秒 .connectTimeout(5, TimeUnit.SECONDS) .build(); ``` #### 2. **Call Timeout 没有启用** 如果希望对整个请求流程进行统一超时控制,必须显式调用 `.callTimeout()` 方法。否则即使设置了 `connectTimeout` 和 `readTimeout`,整体请求仍可能因为某个环节阻塞而超时。 ```java OkHttpClient client = new OkHttpClient.Builder() .callTimeout(10, TimeUnit.SECONDS) // 整个请求最多持续 10 秒 .connectTimeout(5, TimeUnit.SECONDS) .readTimeout(5, TimeUnit.SECONDS) .build(); ``` #### 3. **网络切换导致旧连接未释放** 当设备切换网络(如 Wi-Fi 切换为移动数据)时,OkHttp 默认不会自动关闭之前的连接。此时可能会继续尝试使用旧的连接,导致长时间无响应。 可以通过设置 `connectionPool` 来限制空闲连接的生命周期,减少旧连接残留的影响: ```java OkHttpClient client = new OkHttpClient.Builder() .connectionPool(new ConnectionPool(5, 1, TimeUnit.MINUTES)) .build(); ``` 此外,还可以通过监听网络状态变化并重建 `OkHttpClient` 实例来强制刷新连接。 #### 4. **系统级网络配置干扰** 某些 Android 设备或企业网络环境下,系统级别的 DNS 缓存或代理配置可能影响 OkHttp 的行为。例如,Android 9 及以上版本引入了 `Private DNS` 功能,可能导致 DNS 查询路径不同。 建议在应用层面对 DNS 行为进行完全控制,避免依赖系统默认解析。 #### 5. **使用第三方封装库导致配置失效** 如果使用了 Glide 或 Retrofit 等封装了 OkHttp 的库,需要确保这些库使用的 `OkHttpClient` 实例是你自己配置的那个。例如,在 Glide 中替换默认客户端: ```java public class MyGlideModule extends AppGlideModule { private final OkHttpClient client = new OkHttpClient.Builder() .connectTimeout(5, TimeUnit.SECONDS) .readTimeout(5, TimeUnit.SECONDS) .build(); @Override public void registerComponents(@NonNull Context context, @NonNull Glide glide, @NonNull Registry registry) { registry.replace(GlideUrl.class, InputStream.class, new OkHttpUrlLoader.Factory(client)); } } ``` --- ### 总结 | 问题点 | 解决方案 | |--------|----------| | DNS 解析超时 | 自定义 `Dns` 接口并设置超时 | | 整体请求超时 | 启用 `.callTimeout()` | | 网络切换影响连接 | 使用短生命周期的 `ConnectionPool` | | 系统 DNS 配置干扰 | 显式控制 DNS 查询 | | 第三方库使用默认 Client | 替换为自定义配置的 `OkHttpClient` | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值