新生代:一类是生命周期较短的瞬时对象,这类对象的创建和消亡都非常迅速
注:新生代分为伊甸园区,S0 和S1,
补充:伊甸园区还有一点线程私有的缓冲区
老年代:另外一类对象的生命周期却非常长,在某些极端的情况下还能够与JVM的生命周期 保持一致。
对象分配和收集 1对象分配1.new的对象先放伊甸园区。此区有大小限制。
2.当伊甸园的空间填满时,程序又需要创建对象,JVM的垃圾回收器将对伊甸园区进行垃圾回收(Minor GC),将伊甸园区中的不再被其他对象所引用的对象进行销毁。再加载新的对象放到伊甸园区
3.然后将伊甸园中的剩余对象移动到幸存者0区。
4.如果再次触发垃圾回收,此时上次幸存下来的放到幸存者0区的,如果没有回收,就会放到幸存者1区。
5.如果再次经历垃圾回收,此时会重新放回幸存者0区,接着再去幸存者1区。6.啥时候能去养老区呢?可以设置次数。默认是15次。
可以设置参数:-XX:MaxTenuringThreshold=<N>进行设置。
2对象分配完整流程图 3垃圾回收器(对象收集)
新生代收集(Minon GC / Young GC) :只是新生代(EdenSe,S1)的垃圾收集
老年代收集(Major GC / 01d GC) :只是老年代的垃圾收集。目前,只有CMS GC会有单独收集老年代的行为。
混合收集(Mixed GC):收集整个新生代以及部分老年代的垃圾收集。只有G1和GC有这种行为
整堆收集(Full GC):收集整个java堆和方法区的垃圾收集。
4新生代GC(Minor GC)触发机制:
当年轻代空间不足时,就会触发Minor Gc,这里的年轻代满指的是 Eden代满,Survivor满不会引发GC。(每次young GC会清理年轻代的内存)
因为Java对象大多都具备朝生夕灭的特性,所以young Gc非常频 繁,一般回收速度比较快。
Minor GC会引发STw,暂停其它用户的线程,等垃圾回收结束,用户线 程才恢复运行。
5老年代GC (Major GC/Full GC)触发机制:
出现了Major Gc,经常会伴随至少一次的Minor Gc(但非绝对的,在并行垃圾收集器的收集策略里就有直接进行Major Gc的策略选择过程)
也就是在老年代空间不足时,会先尝试触发Minor Gc。如果之后空间还不足, 则触发Major cc
Major Gc的速度一般会比Minor Gc慢10倍以上,STW的时间更长。如果Major Gc后,内存还不足,就报OOM了。
6Full Gc触发机制(1)调用system.gc()时,系统建议执行Full Gc,但是不必然执行
(2)老年代空间不足
(3)方法区空间不足
(4)通过Minor GC后进入老年代的平均大小大于老年代的可用内存
(5)由Eden区、survivor spacee (From Space)区向survivor space1 (ToSpace)区复制时,对象大小大于To Space可用内存,则把该对象转存到老年 代,且老年代的可用内存小于该对象大小
说明: full gc是开发或调优中尽量要避免的。这样暂时时间会短一些。
7内存分配策略
优先分配到Eden
大对象直接分配到老年代 (尽量避免程序中出现过多的大对象·长期存活的对象分配到老年代)
动态对象年龄判断
注释:如果survivor区中相同年龄的所有对象大小的总和大于survivor空 间的一半,年龄大于或等于该年龄的对象可以直接进入老年代,无须等到MaxTenuringThreshold 中要求的年龄。
空间分配担保 -Xx:HandlePromotionFailure
TLAB(内存缓冲区)为什么有TLAB
堆区是线程共享区域,任何线程都可以访问到堆区中的共享数据
由于对象实例的创建在JVM中非常频繁,因此在并发环境下从堆区中划分内存空间是线程不安全的
为避免多个线程操作同一地址,需要使用加锁等机制,进而影响分配速度。
内存缓冲区简介
从内存模型而不是垃圾收集的角度,对Eden区域继续进行划分,JVM为 每个线程分配了一个私有缓存区域,它包含在Eden空间内。
多线程同时分配内存时,使用TLAB可以避免一系列的非线程安全问题, 同时还能够提升内存分配的吞吐量,因此我们可以将这种内存分配方式称 之为快速分配策略。



