• OKHTTP拦截器:IPv4地址优先连接


    IPV6可能存在的问题:

    起因

    今天,用户反馈 Android 端加载数据较慢,经 Android 开发人员排查后,发现在公司 wifi 下接口响应时间在 50ms 左右,而在 4G 网络下,接口响应时间在 600ms 左右,甚至于 1s 以上,Android 端认为是服务端问题,遂反馈到服务端

    排查

    1. 首先检查 nginx 日志,发现不管是 wifi 网络下还是 4G 网络下,服务端的响应时间均在 30ms 左右,所以排除掉了代码的问题

    2. 向运维同学咨询了一下服务端的网络架构,原来为了 IOS 的 appstore 审核,服务端增加了 ipv6 支持,而 ipv6 的服务入口在美国

    所以怀疑 Android 端解析域名时解析到两个 IP 后,优先使用 IPV6 连接的后端服务

    验证

    使用如下代码,验证 DNS 解析的 IP 地址

    1. try {
    2. InetAddress[] inetAddresses = InetAddress.getAllByName("server.xxxx.cn");
    3. for(InetAddress inetAddress : inetAddresses){
    4. System.out.println(inetAddress.getHostAddress());
    5. }
    6. } catch (UnknownHostException e) {
    7. e.printStackTrace();
    8. }
    1. 连接公司 wifi ,执行代码,只解析到 ipv4 地址
    2. 使用 4G 网络,执行代码,解析到 ipv6 和 ipv4 两个 ip 地址,且 ipv6 一直是首个 ip 地址

    解决

    通过上面的验证,基本断定为 4G 网络下,Android 端通过 ipv6 连接的服务地址。
    端的 http client 库为 okhttp , 查看 javadoc 后,提供了 DNS 接口,代码如下:

    重置Android的线路选择,优先选择ipv4(基于okhttp)

    1. OkHttpClient.Builder builder = new OkHttpClient().newBuilder()
    2. .readTimeout(60, TimeUnit.SECONDS)
    3. .writeTimeout(60, TimeUnit.SECONDS)
    4. .connectTimeout(30, TimeUnit.SECONDS)
    5. .dns(new EngDNS())

    我们通过实现此接口,将解析到的 ip 顺序调整一下,如果是 ipv4 则将其放到数据的第一个,其它保持不变: 

    1. class EngDNS implements Dns {
    2. @Override
    3. public List lookup(String hostname) throws UnknownHostException {
    4. if (TextUtils.isEmpty(hostname)) {
    5. return Dns.SYSTEM.lookup(hostname);
    6. } else {
    7. try {
    8. List inetAddressList = new ArrayList<>();
    9. //获取所有IP地址
    10. InetAddress[] inetAddresses = InetAddress.getAllByName(hostname);
    11. //遍历这里面所有的地址,哪些式IPV4的
    12. for (InetAddress inetAddress : inetAddresses) {
    13. if (inetAddress instanceof Inet4Address) {
    14. inetAddressList.add(0, inetAddress);
    15. } else {
    16. inetAddressList.add(inetAddress);
    17. }
    18. }
    19. return inetAddressList;
    20. } catch (NullPointerException ex) {
    21. return Dns.SYSTEM.lookup(hostname);
    22. }
    23. }
    24. }
    25. }

    再次测试,发现 Android 端的接口响应时间回归正常

    备注

    1. 在测试过程中,发现中国移动和中国电信的 4G 网络 DNS 解析都会解析到两个 IP 地址,而中国联通的 4G 网络只能解析到 ipv4 的址,手机型号是小米 note3
    2. 网上很多的方案都是将 ipv6 关掉,或者在 appstore 审核时打开,审核完成后再关掉,这种不能解决根本问题的方法实在不可取,大家一定要仔细分析,擦亮眼睛
  • 相关阅读:
    云原生事件驱动引擎(RocketMQ-EventBridge)应用场景与技术解析
    计算机视觉: 基于隐式BRDF自编码器的文生三维技术
    如何在指定文件夹下安装python的虚拟环境
    2023计算机毕业设计SSM最新选题之java乡村疫情防控管理系统37804
    Cesium 问题:输出的 纬度 latitude 是 0
    loadrunner-controller-场景执行run
    5 款轻松上手的开源项目「GitHub 热点速览」
    前端vue实现双飞翼布局【flex布局和浮动布局】
    树的入门与习题
    如何入门网络安全有什么条件呢?持有NISP或CISP证书可敲门
  • 原文地址:https://blog.csdn.net/cpcpcp123/article/details/126121183