排查Java进程内存占比过高的方法

 更新时间:2023年10月18日 08:38:49   作者:在下uptown  
某天下午运维反应集成环境的一个Java服务内存飙高,内存耗的太高了,会疑似内存泄漏,所以本文记一次排查Java进程内存占比过高的解决方法,需要的朋友可以参考下

前言

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

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

内存分析

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

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

jmap -histo:live [pid] > a.txt

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

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

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

jstat -gc pid 5000

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

jmap -dump pid

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

解决方案

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

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

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

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

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

到此这篇关于排查Java进程内存占比过高的方法的文章就介绍到这了,更多相关Java进程内存占比过高内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • JAVA 格式化日期、时间的方法

    JAVA 格式化日期、时间的方法

    这篇文章主要介绍了JAVA 格式化日期、时间的方法,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-06-06
  • java8中的lambda表达式,看这篇绝对够

    java8中的lambda表达式,看这篇绝对够

    这篇文章主要介绍了java8中的lambda表达式,看这篇绝对够!具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • Java之一文详解String字符串的用法

    Java之一文详解String字符串的用法

    本文将给大家重点讲解一下String的用法,因为这个太常用,也太常考了。String字符串的内容是比较多的,需要初学者进行专门的学习,尤其是它的一些底层原理更需要我们来了解,需要的同学跟着小编一起学习吧
    2023-05-05
  • 5个JAVA入门必看的经典实例

    5个JAVA入门必看的经典实例

    这篇文章主要为大家详细介绍了5个JAVA入门必看的经典实例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • springboot 重定向方式(redirect前缀)

    springboot 重定向方式(redirect前缀)

    这篇文章主要介绍了springboot 重定向方式(redirect前缀),具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • SpringBoot redis分布式缓存实现过程解析

    SpringBoot redis分布式缓存实现过程解析

    这篇文章主要介绍了SpringBoot redis分布式缓存实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-10-10
  • Java中i++的一些问题总结

    Java中i++的一些问题总结

    这篇文章主要给大家介绍了关于Java中i++的一些问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • 你应该知道的java中的5个注解

    你应该知道的java中的5个注解

    自Java5.0版本引入注解之后,它就成为了Java平台中非常重要的一部分。开发过程中,我们也时常在应用代码中会看到像@Override,@Deprecated这样的注解。下面小编和大家来一起学习一下吧
    2019-05-05
  • Spring Boot 与 kotlin 使用Thymeleaf模板引擎渲染web视图的方法

    Spring Boot 与 kotlin 使用Thymeleaf模板引擎渲染web视图的方法

    这篇文章主要介绍了Spring Boot 与 kotlin 使用Thymeleaf模板引擎渲染web视图的方法,本文给大家介绍的非常详细,具有参考借鉴价值,需要的朋友可以参考下
    2018-01-01
  • Spring定时任务@Scheduled注解(cron表达式fixedRate fixedDelay)

    Spring定时任务@Scheduled注解(cron表达式fixedRate fixedDelay)

    这篇文章主要为大家介绍了Spring定时任务@Scheduled注解(cron表达式fixedRate fixedDelay)使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11

最新评论