记一次排查Java进程内存占比过高🧭

前言

早期面试的时候最害怕面试官问有没有过JVM调优的经验,一听这个问题腿肚子直转筋,JVM八股文都没背多熟练呢,调啥优啊,唠嗑唠的让人害怕,工作一段时间发现不就是基于服务当时所在环境和现象根据JVM现有参数调参嘛,今天记一个JVM内存相关参数。

某天下午运维反应集成环境的一个Java服务内存飙高,看到消息我第一反应就是高就高呗,这不Java本身的特点,高你就多让给它点内存得了,嫌高用Go重构。后来运维说内存耗的太高了,疑似内存泄漏。那得认真分析一波了,别到时候上线发版出了大问题,事关众后端将士的绩效高低,我义不容辞。

内存分析

排查Java进程三板斧,先TOP看资源情况,shit+M按内存排序看是个用户中心的服务,当时看用了接近20G左右,非常高了,集成环境都是研发测试用的环境,没有多少访问量怎么会有这么高的内存使用量。看来多半是泄露了。

分析内存泄露其实也不难,我们先拿到pid浅浅的运行下

less 复制代码
jmap -histo:live [pid] > a.txt

这是获取当前进程中存活的对象统计并把结果输出到a.txt中,从高到低排序,直接看排在前几位的对象有没有我们的业务对象,如果有的话那差不多就逮到了,顺着这个类找一下对应的逻辑分析一下基本就能定位到。但是执行完查询发现排在前几位都是jdk中的class,第一位是c[([代表数组,char[]数组最多也正常,因为String底层就是char[],而且String应用的地方也比较多)

这里已经开始挠头了,不太对啊,按理说内存泄漏肯定是业务中用的类占比会多,再不济也应该是map占用高,这里明显不太符合泄露的现象。。。

分析一下每5s的gc情况发现也都很平稳。

yaml 复制代码
jstat -gc pid 5000

那只能分析dump文件了,可以借助fastThread.io分析dump,也可以搞个JProfile分析。

lua 复制代码
jmap -dump pid

dump之后导入Jprofile发现整体内存占用才2G多,这就很诡异了,从现象上来看像是假装用了20g内存实际只用了2g,这是咋回事呢。

解决方案

本质上是jvm是在启动的时候就会向操作系统申请一部分内存,然后占着自己再做内存管理,当分配对象被回收之后也只是在这个内存区域清除数据然后标记空闲,也就是说根本不会归还给操作系统。根本原因是归还操作系统成本较高,不同的垃圾回收器也有不同的规则。

一直占着也不是个事,JVM还是提供了设置归还策略的参数,MaxHeapFreeRatio ,当空闲区域超过该值时,会进行内存回收,剩余空间的下限为Xms,回收的过程也是线性回收并不是到点下班,到了MaxHeapFreeRatio内存立马降下来。

根据网上大佬们的结论,不同的垃圾回收器下的表现也不一样,详细的结论大家可以自行再去研究。

JAVA 版本 垃圾回收器 参数 是否可以"归还"
JAVA 8 ParallerGC + ParallerOld -Xms100M -Xmx2G -XX:MaxHeapFreeRatio=40
JAVA 8 CMS+ParNew -Xms100M -Xmx2G -XX:MaxHeapFreeRatio=40 -XX:+UseConcMarkSweepGC -XX:+UseParNewGC
JAVA 8 G1 -Xms100M -Xmx2G -XX:MaxHeapFreeRatio=40 -XX:+UseG1GC
JAVA 11 G1 -Xms100M -Xmx2G -XX:MaxHeapFreeRatio=40

根据以上结论将服务启动命令增加了-XX:++UseG1GC -XX:MaxHeapFreeRatio=50(最大空闲比例超过50%归还)参数后内存过高的问题的确没有再出现。

相关推荐
码蜂窝编程官方几秒前
【含开题报告+文档+PPT+源码】基于SpringBoot的线上动物园售票系统设计
java·vue.js·spring boot·后端·spring
Bony-1 小时前
Go语言中值接收者和指针接收者的区别?
开发语言·后端·golang
Cikiss1 小时前
微服务实战——购物车模块实战
java·开发语言·后端·spring·微服务·springcloud
程序猿进阶1 小时前
大循环引起CPU负载过高
java·开发语言·后端·性能优化·并发编程·架构设计·问题排查
ss2732 小时前
被催更了,2025元旦源码继续免费送
java·vue.js·spring boot·后端·微信小程序·开源
Lugas2 小时前
使用Vert.x实现反向代理
java·后端
码农君莫笑2 小时前
在 Blazor 和 ASP.NET Core 中使用依赖注入和Scoped 服务实现数据共享方法详解
前端·后端·c#·.netcore·visual studio
庆 、2 小时前
Django REST framework 源码剖析-视图类详解(Views)
后端·python·django·framework·框架·restful·rest
矩阵猫咪2 小时前
creating-custom-commands-in-flask
后端·python·flask
凡人的AI工具箱2 小时前
每天40分玩转Django:Django Celery
数据库·后端·python·django·sqlite