记一次排查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%归还)参数后内存过高的问题的确没有再出现。

相关推荐
不染_是非39 分钟前
Django学习实战篇六(适合略有基础的新手小白学习)(从0开发项目)
后端·python·学习·django
代码对我眨眼睛1 小时前
springboot从分层到解耦
spring boot·后端
The Straggling Crow1 小时前
go 战略
开发语言·后端·golang
ai安歌1 小时前
【JavaWeb】利用IDEA2024+tomcat10配置web6.0版本搭建JavaWeb开发项目
java·开发语言·后端·tomcat·web·intellij idea
尘浮生2 小时前
Java项目实战II基于Java+Spring Boot+MySQL的作业管理系统设计与实现(源码+数据库+文档)
java·开发语言·数据库·spring boot·后端·mysql·spring
程序员阿鹏3 小时前
ArrayList 与 LinkedList 的区别?
java·开发语言·后端·eclipse·intellij-idea
java_heartLake4 小时前
微服务中间件之Nacos
后端·中间件·nacos·架构
GoFly开发者5 小时前
GoFly快速开发框架/Go语言封装的图像相似性比较插件使用说明
开发语言·后端·golang
苹果酱05675 小时前
通过springcloud gateway优雅的进行springcloud oauth2认证和权限控制
java·开发语言·spring boot·后端·中间件
豌豆花下猫6 小时前
Python 潮流周刊#70:微软 Excel 中的 Python 正式发布!(摘要)
后端·python·ai