Font导致内存泄漏问题排查记录

事件伊始

线上监控告警,某服务进程占用内存超过9G。该服务使用docker运行,且启动命令设置-Xmx为2G。
系统负载内存占用
启动参数
查看GC记录,计算堆内存大小正常
GC堆内存记录
查看内存块
pmap内存信息
判断为内存泄漏问题。

找到问题

使用jmap dump出hprof文件,使用MAT打开进行分析,直接提示潜在内存泄漏问题
MAT
系统中会使用到Font相关的就很快定位到是一个给图片加水印的功能,直接定位到具体方法了。
同时,搜索互联网相关文档信息,找到类似记录:

  1. 记一次Font导致JVM堆外内存泄漏分析
  2. Java使用Font字体的方法

测试环境复现

在测试环境设置docker容器最大可用500M,压测对应接口,持续一段时间后出现无法分配内存错误,容器重启。现象是,开始时每次内存占用接近500M的时候,会进行FullGC, 内存清理后又从400多兆开始慢慢增加。最后到报无法分配内存的时候重启。
容器重启
查看系统日志,已经能够直接定位到代码:
异常栈
再将测试环境docker容器可用内存设置为2G,设置java启动参数-Xmx500M, 压测接口,复现了生产的场景:
测试环境复现

修改代码复测

原代码中,每次请求进入方法,都会使用FontUtil.createFont(resource.getInputStream())创建新的Font对象,修改为使用单例模式,使Font对象为全局唯一,只加载一次。
修改后发布到测试环境再次压测,持续进行较长时间,内存占用稳定,GC次数正常,判断问题解决
请添加图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值