问题重现
开发的一款app,网络访问框架是okhttp。在内网测试,需要设定dns,连接wifi,设置dns,测试为了抓包,wifi设置了代理。问题:首次访问接口,会超时,后续访问接口速度就很快,间隔一段时间不访问,访问接口也会超时,抓包显示,服务器很快就返回了数据,有时候会连续返回两条。
我的okhttp配置
ClearableCookieJar cookieJar1 = new PersistentCookieJar(new SetCookieCache(), new SharedPrefsCookiePersistor(getApplicationContext()));
HttpsUtils.SSLParams sslParams = HttpsUtils.getSslSocketFactory(null, null, null);
// CookieJarImpl cookieJar1 = new CookieJarImpl(new MemoryCookieStore());
OkHttpClient okHttpClient = new OkHttpClient.Builder().connectTimeout(10000L, TimeUnit.MILLISECONDS).readTimeout(10000L, TimeUnit.MILLISECONDS).addInterceptor(new LoggerInterceptor("TAG")).cookieJar(cookieJar1).hostnameVerifier(new HostnameVerifier()
{
@Override
public boolean verify(String hostname, SSLSession session)
{
Log.i("MyApplication", "https, hostname=" + hostname);
return true;
}
}).sslSocketFactory(sslParams.sSLSocketFactory, sslParams.trustManager).build();
OkHttpUtils.initClient(okHttpClient);
分析1:会不会是服务器问题
使用postman访问接口,第一次访问,大概需要120ms,然后后续访问,30ms左右,间隔一段时间访问,也是先120ms,然后30ms,虽然第一次返回比较慢,也没有太夸张,okhttp设置的超时时间是10s,应该不会超时,所以服务器有问题的可能性不大
分析2:会不会是超时时间设置太短
因为会超时,怀疑是okhttp设置的超时时间太短,我设置个比较大的值,60s,
.connectTimeout(60000L, TimeUnit.MILLISECONDS).readTimeout(60000L, TimeUnit.MILLISECONDS)
这个同样会超时,很奇怪,因为服务端最长时间也就130ms,这个分析应该也不成立,和超时时间设置无关
分析3:会不会是重试机制的问题
因为抓包显示,第一次访问接口,服务器时间返回长一些,会调用两次接口,估计是okhttp的重试机制的问题,设置.retryOnConnectionFailure(false)
OkHttpClient okHttpClient = new OkHttpClient.Builder()
.retryOnConnectionFailure(false)
.connectTimeout(DEFAULT_TIMEOUT, TimeUnit.MILLISECONDS)
.build();
这个要把okhttp升级到3.4.1
OKHttp响应超时设置无效
测试结果,还是无效
分析4:会不会和ipv6有关
在网上查资料,看到有的app访问接口,在4g下慢,在wifi下快,这个需要设置dns,okhttp提供了 DNS 接口,通过实现此接口,将解析到的 ip 顺序调整一下,如果是 ipv4 则将其放到数据的第一个
public class ApiDns implements Dns {
@Override
public List<InetAddress> lookup(String hostname) throws UnknownHostException {
if (hostname == null) {
throw new UnknownHostException("hostname == null");
} else {
try {
List<InetAddress> mInetAddressesList = new ArrayList<>();
InetAddress[] mInetAddresses = InetAddress.getAllByName(hostname);
for (InetAddress address : mInetAddresses) {
if (address instanceof Inet4Address) {
mInetAddressesList.add(0, address);
} else {
mInetAddressesList.add(address);
}
}
return mInetAddressesList;
} catch (NullPointerException var4) {
UnknownHostException unknownHostException = new UnknownHostException("Broken system behaviour");
unknownHostException.initCause(var4);
throw unknownHostException;
}
}
}
}
OkHttpClient okHttpClient = new OkHttpClient().newBuilder()
.retryOnConnectionFailure(true)//错误重联
.dns(new ApiDns())
.build();
记一次android 在ipv6访问变慢的解决过程
Android使用手机网络访问慢,但是连接WiFi访问却很快问题解决方案!
可惜,还是不行
分析5:会不会和okhttp版本有关系
因为我使用的okhttp版本比较老,可能在新的版本中解决了这些问题,所以把版本更新了,结果呢,还是不行。
Android 当使用http2时okhttp2 v3.4版本网络连接超时分析
分析6:会不会是okhttp配置的问题
另一款运行良好的app上,也是用的okhttp,所以我把这个接口,在另一个app上访问,结果正常,所以我怀疑是配置问题,然后检查几遍,配置都一样,排除配置问题
分析7:为什么在另一款app上访问正常?
另一款app访问正常,因为已经排除了配置问题,那有可能是测试机器的问题,因为两个app在两台机器上,我把有问题的app放在另一台机器上,访问正常
分析8:机器配置的问题?
因为是网络问题,在两台机器上,一台正常,一台有问题,所以就先判断这两台机器的网络,连接的是同一个wifi,唯一的区别,是有问题的机器,wifi设置过dns,设置过代理,把代理和dns关掉,访问正常
解决
经过以上分析,确定是机器连接wifi,设置了dns的关系,因为访问的内网,所以很有可能是dns影响。折腾了好几天,查了无数资料,试了好几种办法,已经到了绝望的地步,我怀疑服务端问题的时候,还被领导鄙视,说应该让某某某来解决。哎,好在最后解决了,煎熬,记录一下