文章目录
- 前言
- 一、jvm是如何运行代码的?
- 二、jvm的内存模型
- 1 整体内存模型结构图
- 2 堆中的年代区域划分
- 3 对象在内存模型中是如何流转的?
- 4 什么是FULL GC,STW? 为什么会发生FULL GC?
- 5 要调优,首先要知道有哪些垃圾收集器及哪些算法
- 6 调优不是盲目的,要有依据,几款内存诊断工具
- 7 结束语
- 8 出个问题,也是课程中的
- 总结
前言
jvm是java语言可以跨平台运行的基础
jvm 是什么,他是一个可以运行字节码文件的机器;
调优调的是什么?
调整的jvm内存模型中的参数,以及GC垃圾回收器的选择,甚至可以选择使用哪种垃圾回收算法;
那么调优的目的是什么?
调优调的是: 减少GC 的次数,以及GC的STW 时间,这里的GC 大多数指FULL GC
当然minor gc 可能时间会非常长,不过这个情况较为特殊,之后文中会说;
一、jvm是如何运行代码的?
大概步骤为
- java源文件编译为.class文件
- .class文件被各种平台版本的jvm编译为本地机器码(字节码)
- 类加载机制将这些字节码加载后,放到运行时数据中(内存模型中)
- 字节码执行器,通过内存中的入栈出栈执行这些字节码
二、jvm的内存模型
1 整体内存模型结构图
● 堆: 存放对象实例 常量池
● 方法区: 方法信息头,静态变量,常量
● 本地方法栈: native 保留方法运行时候的内存空间
● 程序计数器: 存放执行字节码的行号指示器 (字节码指令的地址)
● java虚拟机栈: 对象的引用,指针,八大基本类型 局部变量
2 堆中的年代区域划分
我们只需要关注我们大多数调整的就好了,那就是年轻代 ,老年代
- 默认新生代 老年代比例 1:2
- 默认新生代中 eden 和 s (s0 s1) 区域的比例为 8:1:1
3 对象在内存模型中是如何流转的?
- 首先new 一个对象的时候,对象一般会在堆中开辟一块内存存储
- 然后这个线程结束,这个对象不再被引用之后,就会被纳入到年轻代,当其中的一块s区域满了,发生轻gc,也就是会发生stw;
- 多种情况会导致对象进入老年代: 例如 一个对象的分代年龄大于15; 对象的整体大小大于s0/s1区域的50% ,不会放入s区,直接进入老年代 等等;
- 分代年龄: 在s区域中的对象,每经过一次轻gc,分代年龄加1;
- 轻gc处理对象的方式: 一部分被加入到老年代,大多数都是从一个s区域复制到另个一s区域(标记复制算法)
- s区域中的两块区域,总有一块是空的;
4 什么是FULL GC,STW? 为什么会发生FULL GC?
- 与轻gc 类似,FULL GC 是发生在老年代的gc
- stw 是 stop the word,所有用户线程都会停止,现象例如: 你在淘宝添加一个物品到购物车,卡住了;
- 发生full gc 与轻gc类似,也就是老年代空间被填满了,必须进行垃圾回收,将无用对象全部移除,释放空间
- 如果释放的空间不够,程序仍然在申请大量的内存,那么此时会发生 oom;
- full gc 一般采用 可达性算法回收(自行百度);
5 要调优,首先要知道有哪些垃圾收集器及哪些算法
- 常用垃圾回收算法汇总
- 常用的垃圾回收器
● Serial 是一个新生代收集器,基于标记-复制算法实现
● Serial Old 是一个老年代收集器,基于标记-整理算法实现
● 两者都是单线程收集,需要「Stop The World」
● Parallel Scavenge 收集器是一款新生代收集器,基于标记-复制算法实现
● Parallel Old 收集器是一款老年代收集器,基于标记-整理算法实现
● 两者都支持多线程并行收集,需要「Stop The World」
● CMS(Concurrent Mark Sweep)是一个老年代收集器,基于标记-清除算法实现
● G1 是一款主要面向服务端应用的垃圾收集器。
● 从整体来看是基于「标记-整理」算法实现的收集器,但从局部(两个 Region 之间)上看又是基于「标记-复制」算法实现
● G1 即是新生代又是老年代收集器",无需组合其他收集器。
可以看到垃圾回收器一般不采用单独的一个算法实现
JDK9 前,我们会在 CMS 和 G1 间选择,对于大概 4GB 到 6GB 以下的堆内存,CMS 一般能处理得比较好,而对于更大的堆内存,可重点考察一下 G1 - 其中G1比较重要,我们详细说下相关参数
-XX:+UseG1GC
启用 G1 垃圾回收器
-XX:InitiatingHeapOccupancyPercent=<45>
当整个 Java 堆的占用达到参数的值时,开始并发标记阶段
-XX:MaxGCPauseMillis=200
G1 暂停时间目标 ( >0 的毫秒数)
-XX:NewRatio=n
新生代与老生代 (new/old generation) 的大小比例 (Ratio). 默认值为 2
-XX:SurvivorRatio=n
Eden/Survivor 空间大小的比例 (Ratio). 默认值为 8
-XX:MaxTenuringThreshold=n
提升年老代的最大临界值 (tenuring threshold). 默认值为 15
-XX:ParallelGCThreads=n
设置垃圾收集器在并行阶段使用的线程数,默认值随 JVM 运行的平台不同而不同
-XX:ConcGCThreads=n
并发垃圾收集器使用的线程数量。 默认值随 JVM 运行的平台不同而不同
-XX:G1ReservePercent=n
作为空闲空间的预留内存百分比,以降低目标空间溢出的风险。默认值是 10%
-XX:G1HeapRegionSize=n
指定每个 Region 的大小。默认值将根据 heap size 算出最优解。最小值为 1Mb, 最大值为 32Mb
- 我们能够调整的参数有哪些(常用参数汇总)
//常见配置汇总
//堆设置
-Xms:初始堆大小
-Xmx:最大堆大小
-XX:NewSize=n:设置年轻代大小
-XX:NewRatio=n:设置年轻代和年老代的比值.如:为3,表示年轻代与年老代比值为1:3,年轻代占整个年轻代年老代和的1/4
-XX:SurvivorRatio=n:年轻代中Eden区与两个Survivor区的比值.注意Survivor区有两个.如:3,表示Eden:Survivor=3:2,一个Survivor区占整个年轻代的1/5
-XX:MaxPermSize=n:设置持久代大小
-XX:MetaspaceSize:设置元空间大小
-XX:MaxMetaspaceSize:设置元空间最大大小
-Xss128k: 设置每个线程的堆栈大小。
//收集器设置
-XX:+UseSerialGC:设置串行收集器
-XX:+UseParallelGC:设置并行收集器
-XX:+UseParalledlOldGC:设置并行年老代收集器
-XX:+UseConcMarkSweepGC:设置并发收集器
//垃圾回收统计信息
-XX:+PrintGC
-XX:+PrintGCDetails
-XX:+PrintGCTimeStamps
-Xloggc:filename
//并行收集器设置
-XX:ParallelGCThreads=n:设置并行收集器收集时使用的CPU数.并行收集//线程数.
-XX:MaxGCPauseMillis=n:设置并行收集最大暂停时间
-XX:GCTimeRatio=n:设置垃圾回收时间占程序运行时间的百分比.公式为1/(1+n)
//并发收集器设置
-XX:+CMSIncrementalMode:设置为增量模式.适用于单CPU情况.
-XX:ParallelGCThreads=n:设置并发收集器年轻代收集方式为并行收集时,使用的CPU数.并行收集线程数.
-XX:+CMSParallelRemarkEnabled:并发清理
6 调优不是盲目的,要有依据,几款内存诊断工具
- jmap jmap使用
- jstack jstack使用
- 阿里arhtas
百度即可;
下载,运行 arthas
● 线程占用过高原因: thread pid
● 死锁信息: thread -b
● 查看当前代码: jad 文件名称
7 结束语
通过看此文章不会让你知道具体怎么调优,但是应该知道如果调优的大体学习路线,具体怎么调优,我是经过自己的学习外加 观看 诸葛老师的课程 传送门
没错,我也是一个毕业于哔哩哔哩的人;
8 出个问题,也是课程中的
调优的目的是为了减少full gc 的次数和时间,尽量通过minor gc处理,这样就可以了,没有问题哈;
问题如下:
我现在有 8核64G的一个服务器,上面运行的程序不是BAT那种级别的,但是也不小;
问题: 堆内存可以设置很大,为了尽量通过轻gc解决,是不是年轻代设置的越大越好?
答案是否定的,因为当年轻代足够大之后,发生minor gc 的时候,也需要stw ,这个时间也会非常久,所以还是需要做适合的大小配置;这也就是我们为什么要调优的原因; 合适的大小才最重要!!!
总结
经过此番学习,对于jvm内存模型,代码运行,对象流转,内存分配有了更高层次认知,对于jvm调优,为什么要调优有清晰的认知,继续学习这篇文章;
改jvm参数不难,难的是你要知道参数的大小的计算; 要明白参数大小的由来