运行时数据区分为方法区、堆区、虚拟机栈,本地方法栈,程序计数器。
jvm的堆区主要存放的是new出来的对象。堆区主要分为老年区、新生代(新生代和老年代比例是1:2)、方法区(逻辑上在堆区)。new创建的对象主要存放在伊甸园区,因此伊甸园区会进行频繁的CG回收。
而新生代分为伊甸园区,form、to(默认比例为8:1:1)这三个区域。
大概结构图如下
堆中GC(GC执行是会触发)
当伊甸园区满的时候会触发伊甸园区的Young GC,会收集年轻代的垃圾,但是form和to区满的时候并不会触发年轻代GC回收(伊甸园区满的时候触发YoungGC)。
对象的分配过程如图
1、创建对象保存在伊甸园区
2、当伊甸园区存满之后会触发GC,然后将存活的对象放到幸存区。
3、当伊甸园区满了之后再次触发GC,然后会将存活下来的对象放入幸存区s0。
4、当伊甸园区再次忙了之后再次触发GC,然后会将伊甸园区和幸存区s0存活下来的对象放入幸存区s1。
5、再次触发GC,会将伊甸园区和幸存区s1中存活下来的对象放入s0区,依次循环。
6、存活的对象被放入一次幸存区那么这个对象的年龄就会加一,当年龄到达一定的次数,该对象就会被放到老年区。
注意:
每次执行GC之后,伊甸园区都是处于一个空的状态。
对象的内存申请流程图大概
为什么要分代?
根据对象存活概论进行分类,将存活时间长的对象放到固定的区域。从而减少GC回收次数和GC回收频率从而提高效率。
为什么有TLAB?
堆区是线程共享的,任何线程都可以访问堆中的共享数据。
由于对象创建是非常频繁的,因此在并发环境下中创建对象,从堆区中划分内存空间是非常不安全的。
为了避免多个线程操作同一个地址,造成安全问题,于是有了加锁等机制,进而影响了分配速率。
什么是TLAB?
JVM为每个线程分配了一个私有缓存区域,它包含在Eden空间内。
多线程分配内存时,使用TLAB可以避免一系列的非线程安全问题。
堆是分配对象储存的唯一选择吗?
堆并不是分配对象储存的唯一选择。当一个对象没有逃逸出方法的时候,就可以被优化为栈上分配,这样就无需在堆中分配空间,也不需要回收了。