解决retrofit OKhttp创建大量对外连接时内存溢出

2024-04-06 12:48

本文主要是介绍解决retrofit OKhttp创建大量对外连接时内存溢出,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这个问题是这样发生的,我的表中有一批数据,量级较大,数百万个,它们有个地址Address字段,标明了地理位置。我需要对这一批数据根据地址去百度或者高德地图去查询经纬度,并且保存下来。
原本是直接分页读取该表,每次读取几百条,然后一条一条去获取经纬度并且保存。后来发现实在太慢,一秒也就能处理个三五条。所以开启了多线程,大约30个线程,每个线程处理不同id范围的数据。
此时问题出现了,每个线程中都有for循环来分页读取DB中的地址数据,然后每条数据都要去百度地图请求一次,网络请求用的是retrofit,retrofit是包装的OKHttp。
这里写图片描述
这里就是构建retrofit的地方,都是一些普通的配置。
然后运行程序后,发现线程数急剧上升,没几秒就跑到了2000多个线程,然后发生内存溢出,程序就挂掉了。
这里写图片描述
这是刚启动项目时,线程数只有50多个。当开启多线程任务后
这里写图片描述
可以看到线程数干到2000多时程序崩了

java.lang.OutOfMemoryError: unable to create new native threadat java.lang.Thread.start0(Native Method)at java.lang.Thread.start(Thread.java:714)at java.util.concurrent.ThreadPoolExecutor.addWorker(ThreadPoolExecutor.java:950)at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1368)at okhttp3.ConnectionPool.put(ConnectionPool.java:153)at okhttp3.OkHttpClient$1.put(OkHttpClient.java:163)at okhttp3.internal.connection.StreamAllocation.findConnection(StreamAllocation.java:201)at okhttp3.internal.connection.StreamAllocation.findHealthyConnection(StreamAllocation.java:121)at okhttp3.internal.connection.StreamAllocation.newStream(StreamAllocation.java:100)at okhttp3.internal.connection.ConnectInterceptor.intercept(ConnectInterceptor.java:42)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:92)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:67)at okhttp3.internal.cache.CacheInterceptor.intercept(CacheInterceptor.java:93)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:92)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:67)at okhttp3.internal.http.BridgeInterceptor.intercept(BridgeInterceptor.java:93)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:92)at okhttp3.internal.http.RetryAndFollowUpInterceptor.intercept(RetryAndFollowUpInterceptor.java:120)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:92)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:67)at com.mindata.ecserver.global.http.RetrofitServiceBuilder.lambda$generateClient$0(RetrofitServiceBuilder.java:72)at com.mindata.ecserver.global.http.RetrofitServiceBuilder$$Lambda$11/620324267.intercept(Unknown Source)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:92)at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.java:67)at okhttp3.RealCall.getResponseWithInterceptorChain(RealCall.java:185)at okhttp3.RealCall.execute(RealCall.java:69)at retrofit2.OkHttpCall.execute(OkHttpCall.java:180)at com.mindata.ecserver.global.http.CallManager.execute(CallManager.java:25)

通过监控界面可以看到,大量的线程OKHttp ConnectionPool,也是导致内存溢出的主要原因。
这里写图片描述
为什么会有这么多的OkHttp ConnectionPool呢。通过分析代码发现,我有30个线程,每个线程里会分配数万的数据来进行百度地图的请求,每次循环,都会发起一个retrofit网络请求去访问百度,这样很快就有数千个http请求出去了。
伪代码大概是这样的

public BaiduCoordinateService getBaiduCoordinateService(RequestProperty requestProperty) {return generateRetrofit(requestProperty).create(BaiduCoordinateService.class);}这是构建百度地图请求的,我的用法是for(int i = 0; i < 5000; i++) {getBaiduCoordinateService(xxx).fetchBaiduCoordinate();}

网络请求多没什么,关键是我的每个请求得到结果后就不用了,但是系统依旧保持了这个请求的线程没有释放,直接导致线程池越来越大,很快超过最大限制就崩溃了。
通过各方查证,有的说,请求header里的Connection,写的是keep alive,导致了长连接,所以我把构建retrofit的header的地方改成了Connection为close,然而没什么卵用。依旧是上面的问题,很快线程数超过就崩溃了。
后来开始调查OkHttpClient的ConnectionPool,这个就是OkHttp网络请求的线程池,在OkHttpClient源码中可以看到

public OkHttpClient.Builder connectionPool(ConnectionPool connectionPool) {if (connectionPool == null) {throw new NullPointerException("connectionPool == null");} else {this.connectionPool = connectionPool;return this;}}

在OkHttpClient的源码中,默认的构造方法里可以看到默认最大线程空闲数是5,keepAlive时间为5分钟。也就是发起一次网络连接后,5分钟内不会断开连接。
这里写图片描述
那么问题就出在这里了,我在短时间内发起了大量网络连接,每个是一个线程,而且每个都默认保存5分钟,很快线程数就超标了。
考虑到我的每次请求都是一次性的,所以我修改了ConnectionPool的keepAliveDuration时间,让每次连接1秒后就关闭。
这里写图片描述
之后再次运行程序,发现OK了,线程数最大也没超过200,程序也没再抛出过outofmemery异常。
后来又仔细回想了一下,发现哪里怪怪的,为毛我会有这么多的ConnectionPool连接呢?
又回头看了一下创建retrofit的请求service的地方,发现了最大的问题所在,我在对数据库循环过程中,每条数据都创建了一个service,如上面的伪代码那里所写,每一条数据都走了一遍generateClient。这才是罪魁祸首!因为我的项目中,多个地方会使用retrofit,会创建baseUri不同的请求,所以我想成了每次请求都创建一个客户端的方式。没想到这样会创建一个额外的线程。
最终解决方式是,我又把设置OkHttp5分钟那里给恢复默认了,然后对于baidu的请求,只创建一个service,而不是在循环里去创建多个retrofit客户端。
最后再次运行,这下线程更少了,只剩90个了。

这篇关于解决retrofit OKhttp创建大量对外连接时内存溢出的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/879922

相关文章

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

oracle数据库索引失效的问题及解决

《oracle数据库索引失效的问题及解决》本文总结了在Oracle数据库中索引失效的一些常见场景,包括使用isnull、isnotnull、!=、、、函数处理、like前置%查询以及范围索引和等值索引... 目录oracle数据库索引失效问题场景环境索引失效情况及验证结论一结论二结论三结论四结论五总结ora

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

element-ui下拉输入框+resetFields无法回显的问题解决

《element-ui下拉输入框+resetFields无法回显的问题解决》本文主要介绍了在使用ElementUI的下拉输入框时,点击重置按钮后输入框无法回显数据的问题,具有一定的参考价值,感兴趣的... 目录描述原因问题重现解决方案方法一方法二总结描述第一次进入页面,不做任何操作,点击重置按钮,再进行下

Mysql 中的多表连接和连接类型详解

《Mysql中的多表连接和连接类型详解》这篇文章详细介绍了MySQL中的多表连接及其各种类型,包括内连接、左连接、右连接、全外连接、自连接和交叉连接,通过这些连接方式,可以将分散在不同表中的相关数据... 目录什么是多表连接?1. 内连接(INNER JOIN)2. 左连接(LEFT JOIN 或 LEFT

解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题

《解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题》本文主要讲述了在使用MyBatis和MyBatis-Plus时遇到的绑定异常... 目录myBATis-plus-boot-starpythonter与mybatis-spring-b

关于Java内存访问重排序的研究

《关于Java内存访问重排序的研究》文章主要介绍了重排序现象及其在多线程编程中的影响,包括内存可见性问题和Java内存模型中对重排序的规则... 目录什么是重排序重排序图解重排序实验as-if-serial语义内存访问重排序与内存可见性内存访问重排序与Java内存模型重排序示意表内存屏障内存屏障示意表Int

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择

电脑显示hdmi无信号怎么办? 电脑显示器无信号的终极解决指南

《电脑显示hdmi无信号怎么办?电脑显示器无信号的终极解决指南》HDMI无信号的问题却让人头疼不已,遇到这种情况该怎么办?针对这种情况,我们可以采取一系列步骤来逐一排查并解决问题,以下是详细的方法... 无论你是试图为笔记本电脑设置多个显示器还是使用外部显示器,都可能会弹出“无HDMI信号”错误。此消息可能

mysql主从及遇到的问题解决

《mysql主从及遇到的问题解决》本文详细介绍了如何使用Docker配置MySQL主从复制,首先创建了两个文件夹并分别配置了`my.cnf`文件,通过执行脚本启动容器并配置好主从关系,文中还提到了一些... 目录mysql主从及遇到问题解决遇到的问题说明总结mysql主从及遇到问题解决1.基于mysql