堆
一、堆的概述
1.1 介绍
- 一个JVM实例只存在一个堆内存,堆也是java内存管理的核心区域
- java堆区在jvm启动的时候即被创建,其空间大小也就确定了。是jvm管理的最大的内存空间:
- 堆内存的大小是可以调节的
- 《java虚拟机规范》规定,堆可以处于物理上不连续的内存空间中,但在逻辑上它应该被视为连续的
- 所有的线程共享java堆,在这里可以划分线程私有的缓冲区(Thread Local Alllcation Buffer,TLAB)
- 几乎所有的对象实例以及数组都应当在运行时分配在堆上
- 数组和对象可能永远不会存储在栈上,因为栈帧保存引用,这个引用指向对象或者数组在堆中的位置
- 在方法结束后,堆中的对象不会马上被移除,仅仅在垃圾回收的时候才会被移除
- 堆,是GC(Garbage Collection 垃圾收集器)执行垃圾回收的重点区域
1.2 栈-堆-方法区 关系图
1.3 堆空间的内存细分
1.4 堆空间内部结构
年轻代、老年代、永久代;主要讲新生代和养老带;
二、设置堆内存大小与OOM
2.1 设置堆空间大小的参数
- java堆区用于存储java对象实例,堆的大小在jvm启动时就已经设定好了
1.“-Xms”用于表示堆区的起始内存;
-X是jvm的运行参数
ms 是memory start
2.“-Xmx”用于表示堆去的最大内存;
注:开发中建议将初始堆内存和最大堆内存设置成相同的值
2.2 默认堆空间的大小
1.初始内存大小:物理电脑内存大小/64
2.最大内存大小:物理电脑内存大小/4
2.3 OOM(OutOfMemory)
栈内存溢出
三、年轻代与老年代
3.1 介绍
1.存储在jvm中的java对象可以被划分两类:
1).一类是生命周期较短的瞬时对象,这类对象的创建和消亡都非常迅速
2).另外一类的声明周期却非常长,在某些极端的情况下还能够与jvm的生命周期保持一致
2.java堆区进一步细分的话,可以划分为年轻代(YoungGen)和老年代(OldGen)
3.其中年轻代又可以分为Eden空间,Survivor0空间和Survivor1空间(有时也叫from区、to区)
3.2 图解:
3.3 空间占比
- 新生代和老年代在堆结构中的空间占比
NewRatio表示老年代与新生代的比例;
1. 默认-XX:NewRatio=2,表示新生代占1,老年代占2,新生代占整个堆的1/3;
2. 可以修改-XX:NewRatio=4,表示新生代占1,老年代占4,新生代占整个堆的1/5;
- 这些参数开发中一般不会调:
- 一些设置参数
1.几乎所有的java对象都是在Eden(伊甸园)被new出来
2.绝大部分的java对象的销毁都在新生代进行了
四、图解对象分配过程
4.1 对象分配过程
1.new的对象先放伊甸园区;
2.伊甸园的空间填满时对伊甸园区进行垃圾回收;
3.将伊甸园剩余的对象移动到幸存者0区;
4.再次触发垃圾回收时,上次幸存下来的对象,如果没有回收,将其放到幸存者1区,标记age为1
5.如果再次经历垃圾回收,会重新放回幸存者0区,接着再去幸存者1区。age为15时,晋升到养老区。
4.2 对象分配的特殊情况:
当创建超大对象时,如果伊甸园区放不下,则直接放入老年代区,如果老年代区也放不下,则报OOM
总结:
1.针对幸存者S0,S1区的总结:复制之后有交换,谁空谁是to;
2.关于垃圾回收,频繁在新生区收集,很少在养老区收集,几乎不在永久区/元空间收集。
五、Minor GC、Major GC、Full GC
5.1 介绍
jvm在进行GC时,并非每次都对上面三种内存(新生代、老年代;方法区)一起进行回收的,大部分时候回收的都是指新生代
- GC按照回收区域分为两种类型:
1.部分收集:不是完整收集整个java堆的垃圾收集。其中又分- 新生代收集(Minor GC/ Young GC)
- 老年代收集(Major GC/ Old GC)
- 目前,只有CMS GC 会有单独收集老年代的行为
- 注意,很多时候Major GC会和Full GC混淆使用,需要具体分辨是老年代回收还是整堆回收
- 混合收集(Mixed GC):收集整个新生代和部分老年代的垃圾收集
- 目前,只有G1 GC 会有这种行为
- 整堆收集(Full GC):收集整个java堆和方法区的垃圾收集
5.2 Minor GC触发机制
5.3 Major GC触发机制
5.4 Full GC触发机制
六、堆空间分代思想
6.1 为什么要进行java堆分代
- 图1
- 图2
七、内存分配策略
八、为对象分配内存:TLAB
8.1 为什么有TLAN
8.2 什么是TLAB
九、小结堆空间的参数设置
十、堆是分配对象的唯一选择吗?
1.堆参数
本章小结: