Okhttp连接泄漏警告问题分析
创始人
2024-04-11 01:36:56
0

背景

  某天在查询生产日志时,发现大量的Okhttp连接泄漏警告日志,但生产上没有收到任何异常反馈。出于好奇心,本地最小化复现问题,并最终解决问题。
在这里插入图片描述

分析问题

  1. okhttp官网的demo示例

    OkHttpClient client = new OkHttpClient();String run(String url) throws IOException {Request request = new Request.Builder().url(url).build();try (Response response = client.newCall(request).execute()) {return response.body().string();}
    }
    

    1.1 上述示例明显是Try-with-resources写法,最终会在finally里去关闭资源
    1.2 跟进string()源码发现最终也会关闭资源

  2. 分析官网示例,连接泄漏可能是因为执行请求没有关闭资源并且也没有执行string方法,导致资源始终没有关闭
    在这里插入图片描述
    2.1 上面示例在响应成功时就不会关闭连接资源

复现问题

  1. 启动一个简单服务端,暴露一个接口供客户端访问
    在这里插入图片描述
  2. 客户端请求
    在这里插入图片描述
    2.1 在执行请求时,会将调用者添加连接信息中(一个连接包含多个调用者),并且将调用者用虚引用包装
    在这里插入图片描述
    2.2 点进CallReference发现调用者用虚引用包装
    在这里插入图片描述
    2.3 在sendLocalhost方法执行完后,调用者引用关闭等待垃圾回收(当垃圾回收后,虚引用里的实际对象变为Null)
    (1)执行完sendLocalhost方法后打印断点,等待RealConnectionPool#pruneAndGetAllocationCount检查执行 在这里插入图片描述
    (2)在执行方法1的时候之前打上断点,通过System.gc()主动触发GC回收没有引用的调用者,这时reference.get()返回null,复现线上告警日志(为了让其触发垃圾回收,我把堆栈调的比较小)
    在这里插入图片描述
    在这里插入图片描述
    2.4 当连接的所有调用者都没有之后,进行连接驱逐释放

总结

  1. 搜索代码,将所有没有关闭资源的地方都进行关闭。后续发现再也没有相关警告日志出现了,成功解决问题
  2. 规范代码,调用请求和获取响应作为整体不进行拆分,然后将响应进行类型转换

相关内容

热门资讯

监控摄像头接入GB28181平... 流程简介将监控摄像头的视频在网站和APP中直播,要解决的几个问题是:1&...
Windows10添加群晖磁盘... 在使用群晖NAS时,我们需要通过本地映射的方式把NAS映射成本地的一块磁盘使用。 通过...
protocol buffer... 目录 目录 什么是protocol buffer 1.protobuf 1.1安装  1.2使用...
在Word、WPS中插入AxM... 引言 我最近需要写一些文章,在排版时发现AxMath插入的公式竟然会导致行间距异常&#...
Fluent中创建监测点 1 概述某些仿真问题,需要创建监测点,用于获取空间定点的数据࿰...
educoder数据结构与算法...                                                   ...
MySQL下载和安装(Wind... 前言:刚换了一台电脑,里面所有东西都需要重新配置,习惯了所...
MFC文件操作  MFC提供了一个文件操作的基类CFile,这个类提供了一个没有缓存的二进制格式的磁盘...
有效的括号 一、题目 给定一个只包括 '(',')','{','}'...
【PdgCntEditor】解... 一、问题背景 大部分的图书对应的PDF,目录中的页码并非PDF中直接索引的页码...