点赞再看,养成习惯,微信搜索「小大白日志」关注这个搬砖人。

文章不定期同步公众号,还有各种一线大厂面试原题、我的学习系列笔记。

基础概念

  • 一般JVM调优,重点在于调整JVM堆大小、调整垃圾回收器
  • jvm调优的目的是,减少full gc、降低gc停顿时间、提高吞吐量;调优的顺序=“提高吞吐量”>“降低gc停顿时间”;在满足吞吐量的前提下,再降低gc停顿时间;若不能同时满足以上,则选择最适合系统的一种调优结果

JVM调优常用参数

  • -Xmx1024m:最大堆内存,当物理内存不超过192m时最大堆内存为物理内存的一半,否则为物理内存的四分一
  • -Xms1024m:最小堆内存,一般设置为与-Xmx同等值
  • -XX:+PrintGCDetails:输出gc详细日志
  • -XX:+PrintGCTimeStamps或-XX:+PrintGCDateStamps:输出gc信息时带上时间戳
  • jcmd:专用于查看JVM状态,可以查看正在运行的进程,会显示出进程号

  • jmap -heap 10864:查看进程号为10864的进程的堆使用情况

具体参数如下:

F:\Iwebapp\demo-test>jmap -heap 10864
Attaching to process ID 10864, please wait...
Debugger attached successfully.
Server compiler detected.
JVM version is 25.65-b01 using thread-local object allocation.
Parallel GC with 4 thread(s) Heap Configuration:
MinHeapFreeRatio = 0 //空余堆内存最小百分比,HeapFreeRatio=空闲堆内存的百分比=现有空闲堆内存/总共对内存*100,当MinHeapFreeRatio<HeapFreeRatio时,在每次垃圾回收之后需要进行对扩容
MaxHeapFreeRatio = 100 //空余堆内存最大百分比
MaxHeapSize = 1044381696 (996.0MB) //996.0MB=最大堆内存无物理内存1/4=4G*1/4约等于1G
NewSize = 22020096 (21.0MB) //年轻代堆内存的初始大小
MaxNewSize = 348127232 (332.0MB) //年轻代堆内存的允许最大值,因为年轻代:老年代=1:2,所以年轻代最大=996*1/3=332
OldSize = 45088768 (43.0MB)
NewRatio = 2 //老年代:年轻代=2:1
SurvivorRatio = 8 //年轻代中Eden区:Survivor区=8:2,Survivor区=SurvivorFrom区+SurvivorTo区=1+1=2
MetaspaceSize = 21807104 (20.796875MB) //jdk1.8之后使用的元空间默认大小
CompressedClassSpaceSize = 1073741824 (1024.0MB)
MaxMetaspaceSize = 17592186044415 MB //jdk1.8之后使用的元空间的最大值
G1HeapRegionSize = 0 (0.0MB) //若使用G1垃圾回收器,JVM把堆内存每个多个大小相等的区域,指定每个区域的大小 Heap Usage:
PS Young Generation
Eden Space: //年轻代的使用情况
capacity = 289931264 (276.5MB)
used = 160612016 (153.1715545654297MB)
free = 129319248 (123.32844543457031MB)
55.39658392963099% used
From Space: //SurvivorFrom区的使用情况
capacity = 5767168 (5.5MB)
used = 65536 (0.0625MB)
free = 5701632 (5.4375MB)
1.1363636363636365% used
To Space: //SurvivorTo区的使用情况
capacity = 5242880 (5.0MB)
used = 0 (0.0MB)
free = 5242880 (5.0MB)
0.0% used
PS Old Generation //老年代的使用情况
capacity = 37748736 (36.0MB)
used = 13700064 (13.065399169921875MB)
free = 24048672 (22.934600830078125MB)
36.29277547200521% used

监控JVM

实战源码:以demo-test工程为例,用gcviewner和jdk内置的监控jvm命令来做jvm调优

package com.example.demo;
import com.example.util.ThreadTest;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication
public class DemoTestApplication { public static void main(String[] args) {
SpringApplication.run(DemoTestApplication.class, args);
ThreadTest.testJVM();
} } public class ThreadTest { public static void main(String[] args) {
testJVM();
} public static void testJVM(){
//匿名内部类重写Runnable的run方法可以写成箭头函数:()->{run方法的内容}
//1000ms后每50ms执行创建1个线程,每个线程150*512*1024byte=150*512*1024B=150*512k=150*0.5M=75M,1s内最多1000/50=20个线程,则1s内最大占用内存=75*20=150M
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(()->{
new Thread(()->{
for(int i=1;i<=150;i++){
byte[] b=new byte[1024*512];//0.5M
}
System.out.println(Thread.currentThread().getName()+"已分配512k内存...");
}).start();
},1000,50,TimeUnit.MILLISECONDS);
}

在target目录下【java -jar -Xloggc:gc.log -Xmx35m -Xms35m -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintGCDateStamps demo-test.jar】启动jar包或者直接在idea中添加【-Xloggc:gc.log -Xmx35m -Xms35m -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintGCDateStamps】启动springboot启动类,以下用gcviewner监控JVM :

  • gcviwner

gcviwner是一款可以监控GC日志的工具,前提是要拿到gc日志,gcviwner用法可参考【https://github.com/chewiebug/GCViewer】 ,命令 【java -jar gcviewer-1.37-SNAPSHOT.jar】启动gcviwner,并导入gc.log日志,如下,

上中下三张图,由上图可知吞吐量为74.26%,一般可以达到90%以上,红线是堆的实时总内存,蓝线是堆的实时已使用内存,灰色为young gc,黑色为full gc,可知灰色的线太密集=young gc频繁,黑色的线也是太密集=full gc频繁;full gc总数为296,一般可以把full gc降到个位数;

由中图可知最大停顿时间比较大,达到了892ms,一般可调优到几十毫秒级别;

有下图可看到young gc和full gc的情况,full gc次数296,比较频繁;同时元空间metadata区域过小,可以适当调大

由上,把metadata调为64M,执行【java -jar -XX:MetaspaceSize=64m -Xloggc:gc.log -Xmx35m -Xms35m -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintGCDateStamps demo-test.jar】重新启动jar包:

可知full gc次数还是比较多、最大停顿时间也比较长、吞吐量也低,尝试着增大'最大堆内存',执行【java -jar -XX:MetaspaceSize=64m -Xloggc:gc.log -Xmx100m -Xms100m -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintGCDateStamps demo-test.jar】重新启动jar包:

可知full gc次数变少了(黑线少了,0次)、最大停顿时间也短了(94.65ms,到了十几毫秒级别)、吞吐量也高了(93.83%)

至此,调优基本满足上面的【减少full gc、降低gc停顿时间、提高吞吐量】的要求。

下面是jdk内置的jvm监控命令

  • jstat命令

jstat命令用于监视虚拟机的运行状态,命令格式【jstat -参数命令 进程号 间隔毫秒数 总输出次数】,主要参数项:

//每1000ms统计一次demo-test-0.0.1.jar的gc信息
jstat -gc $(jcmd | grep "demo-test-0.0.1.jar" | awk '{print $1}') 1000



S0C、S1C、S0U、S1U:C是总量,U是使用量,代表S0(survivorFrom区)和S1(survivorTo区)的总量与使用量

EC、EU:Eden区总量与使用量

OC、OU:Old区总量与使用量

MC、MU: Metaspace区(元空间)总量与使用量

YGC、YGCT:young gc的次数与时间

FGC、FGCT:full gc的次数与时间<

GCT:总的GC时间

//每1000ms统计一次进程ID为10492的gc信息,共统计10次数结束统计
jstat -gc 10492 10

  • jps

jps可以查看正在运行的java进程,命令格式【jsp 参数】,主要参数项:

【jsp -l】查看具体的包名

  • jinfo

可以查看启动jar包时未显式指定的系统默认值,以及动态修改这些默认值,命令格式【jinfo -flag 默认的参数 进程id】,查看【年轻代:老年代】的内存比例:

  • jmap

jmap查看java内存信息,命令格式【jmap -参数 进程id】,参数项:

实例如上:【jmap -heap 10864】查看进程号为10864的进程的堆使用情况
  • jstack

jstack是一个非常有用的命令,可以查看正在运行的线程的堆栈信息,比如查看后天没有相应的线程在做些什么,在等待什么资源,命令格式【jstack 参数项 进程号】,主要参数项如下:

//查看进程的堆栈及锁的信息
jstack -l 10220



图中线程状态是很重要的一个要素:

CMS收集器和G1收集器的调优

启动各种垃圾收集器
参数 说明
-XX:+UseParallelGC 新生代使用并行回收收集器
-XX:+Use ParalleloldGC 老年代使用并行回收收集器
-XX:ParallelGCThreads 设置用于垃圾回收的线程数
-XX:+UseAdaptiveSizePolicy 打开自适应GC策略
CMS和G1的调优
  • cms是分代收集算法中老年代的算法(用标记整理算法),G1却适用于年轻代和老年代,故G1可以采用复制算法不会导致内存碎片,而cms不能采用年轻代的复制算法而导致有内存碎片
  • G1是由一个个区域块组成的,即使有碎片也只会影响某个区域块,不会影响整个堆
  • cms采用老年代标记整理算法,需要扫描整个表来标记整理,故存在停顿(即使它是并发);G1的停顿取决于需要收集的'区域块'个数,而不用等整个堆空间,所以g1的停顿时间比cms少

OK,如果文章哪里有错误或不足,欢迎各位留言。

创作不易,各位的「三连」是二少创作的最大动力!我们下期见!

JVM调优篇的更多相关文章

  1. jvm系列(七):jvm调优-工具篇

    16年的时候花了一些时间整理了一些关于jvm的介绍文章,到现在回顾起来还是一些还没有补充全面,其中就包括如何利用工具来监控调优前后的性能变化.工具做为图形化界面来展示更能直观的发现问题,另一方面一些耗 ...

  2. JVM调优-工具篇

    原文地址 16年的时候花了一些时间整理了一些关于jvm的介绍文章,到现在回顾起来还是一些还没有补充全面,其中就包括如何利用工具来监控调优前后的性能变化.工具做为图形化界面来展示更能直观的发现问题,另一 ...

  3. <JVM下篇:性能监控与调优篇>补充:使用OQL语言查询对象信息

    笔记来源:尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) 同步更新:https://gitee.com/vectorx/NOTE_JVM https://codechina.cs ...

  4. <JVM下篇:性能监控与调优篇>补充:浅堆深堆与内存泄露

    笔记来源:尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) 同步更新:https://gitee.com/vectorx/NOTE_JVM https://codechina.cs ...

  5. <JVM下篇:性能监控与调优篇>03-JVM监控及诊断工具-GUI篇

    笔记来源:尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) 同步更新:https://gitee.com/vectorx/NOTE_JVM https://codechina.cs ...

  6. <JVM下篇:性能监控与调优篇>01-概述篇-02-JVM监控及诊断工具-命令行篇

    笔记来源:尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) 同步更新:https://gitee.com/vectorx/NOTE_JVM https://codechina.cs ...

  7. Jvm调优理论篇

    Jvm实战调优 OOM(Out Of Memory) 内存溢出错误 ps:由于Java虚拟机有许多实现,本文主要阐述的是OpenJDK的HotSpot虚拟机,JDK版本是8. 一.首先要明白造成OOM ...

  8. 阿里面试100%问到,JVM性能调优篇

    JVM 调优概述 性能定义 吞吐量 - 指不考虑 GC 引起的停顿时间或内存消耗,垃圾收集器能支撑应用达到的最高性能指标. 延迟 - 其度量标准是缩短由于垃圾啊收集引起的停顿时间或者完全消除因垃圾收集 ...

  9. jvm系列(四):jvm调优-命令篇

    运用jvm自带的命令可以方便的在生产监控和打印堆栈的日志信息帮忙我们来定位问题!虽然jvm调优成熟的工具已经有很多:jconsole.大名鼎鼎的VisualVM,IBM的Memory Analyzer ...

随机推荐

  1. Kruscal algorithm

    #include <iostream> #include <algorithm> using namespace std; #define MAX 5 #define INF ...

  2. fiber核心(react 16)?

    旧: 浏览器渲染引擎单线程, 计算DOM树时锁住整个线程, 所有行为同步发生, 有效率问题, 期间react会一直占用浏览器主线程,如果组件层级比较深,相应的堆栈也会很深,长时间占用浏览器主线程, 任 ...

  3. JDBC中大数据量的分页解决方法?

    最好的办法是利用sql语句进行分页,这样每次查询出的结果集中就只包含某页的数据内容. sql语句分页,不同的数据库下的分页方案各不一样,下面是主流的三种数据库的分页sql: oracle: selec ...

  4. 学习GlusterFS(五)

    一,分布式文件系统理论基础 1.1 分布式文件系统出现 计算机通过文件系统管理,存储数据,而现在数据信息爆炸的时代中人们可以获取的数据成指数倍的增长,单纯通过增加硬盘个数来扩展计算机文件系统的存储容量 ...

  5. 学习heartbeat-04 原理及部署

    1. Heartbeat介绍 1.1 Heartbeat作用 通过它可以将资源(IP及程序服务等资源)从一台故障计算机快速转移到另一台运转正常的机器继续提供服务,在实际生产应用场景中,heartbea ...

  6. C语言对源程序处理的四个步骤:预处理、编译、汇编、链接——预处理篇

    预处理 1)预处理的基本概念 C语言对源程序处理的四个步骤:预处理.编译.汇编.链接. 预处理是在程序源代码被编译之前,由预处理器(Preprocessor)对程序源代码进行的处理.这个过程并不对程序 ...

  7. C++ pair的基本用法总结

    1,pair的应用 pair是将2个数据组合成一组数据,当需要这样的需求时就可以使用pair,如stl中的map就是将key和value放在一起来保存.另一个应用是,当一个函数需要返回2个数据的时候, ...

  8. MATLAB quadprog函数求解二次规划问题

                     [例]求如下二次规划问题. [分析]首先应该把目标函数表示成如下矩阵形式: 这里要细说一下如何写成矩阵形式. 首先,向量x是很容易写出的,因为f(x)包含两个变量x1 ...

  9. C++重载输入流、输出流运算符

    在c++中类的私有成员是不能被直接访问的,需要通过类中提供的成员函数简介的操作这些数据.同时C++ 能够使用流提取运算符 >> 和流插入运算符 << 来输入和输出内置的数据类型 ...

  10. PCB布线总的原则

    转自张飞实战电子公众号 PCB布线总的原则 最短路径和减少干扰 PCB布线的总的流程大致如下: 1了解制造厂商的制造规范-线宽,线间距,过孔要求及层数要求: 2确定层数并定义各层的功能: 3设计布线规 ...