Okhttp首次访问接口超时的分析以及解决办法

问题重现

开发的一款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影响。折腾了好几天,查了无数资料,试了好几种办法,已经到了绝望的地步,我怀疑服务端问题的时候,还被领导鄙视,说应该让某某某来解决。哎,好在最后解决了,煎熬,记录一下

  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 11
    评论
您好!要使用 OkHttp 调用 WebService 接口,您可以按照以下步骤进行操作: 1. 添加 OkHttp 依赖:将以下代码添加到您的项目的 build.gradle 文件中的 dependencies 部分。 ```groovy implementation 'com.squareup.okhttp3:okhttp:4.9.0' ``` 2. 创建 OkHttp 客户端:您可以使用以下代码创建一个 OkHttp 客户端。 ```java OkHttpClient client = new OkHttpClient(); ``` 3. 构建 SOAP 请求消息体:构建符合 WebService 要求的 SOAP 请求消息体。这通常涉及创建一个包含所需参数的 XML 文档。 ```java String requestBody = "<soap:Envelope xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">\n" + " <soap:Body>\n" + " <YourRequestTag>\n" + " <param1>Value1</param1>\n" + " <param2>Value2</param2>\n" + " </YourRequestTag>\n" + " </soap:Body>\n" + "</soap:Envelope>"; ``` 4. 创建 HTTP 请求:使用 OkHttp 创建一个 POST 请求,并设置请求的 URL、头信息和请求体。 ```java Request request = new Request.Builder() .url("http://example.com/your-webservice-endpoint") .header("Content-Type", "text/xml;charset=utf-8") .post(RequestBody.create(MediaType.parse("text/xml"), requestBody)) .build(); ``` 5. 发送请求并处理响应:使用 OkHttpOkHttpClient 执行请求,并处理响应。 ```java try (Response response = client.newCall(request).execute()) { if (response.isSuccessful()) { String responseBody = response.body().string(); // 处理响应数据 } else { // 处理错误情况 } } catch (IOException e) { e.printStackTrace(); } ``` 请根据您实际的 WebService 接口定义和参数进行适当的修改。这只是一个基本的示例,具体的实现可能因接口规范而有所不同。希望这可以帮助到您!如果您有任何其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值