Java堆(Java Heap)是Java虚拟机所管理的内存中最大的一块。
Java堆是被所有线程共享的一块区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。
但随着JIT编译器的发展和逃逸分析技术的逐渐成熟,栈上分配、标量替换优化技术会导致不在堆上分配对象成为可能。
Java堆是垃圾收集器管理的主要区域,因此很多时候Java堆也被称为“GC”堆。

new的对象先放在伊甸园区,此区有大小限制。
当伊甸园的空间填满时,程序又需要创建对象,JVM垃圾货回收器将对伊甸园区进行垃圾回收(Minor GC或者Young GC),将伊甸园区不在被其他对象所引用的对象进行销毁。在加载新的对象放到伊甸园区。
然后将伊甸园区没有回收的对象年龄加1放到幸存者0区
如果再次触发垃圾回收,会对伊甸园区和幸存者0区的对象进行垃圾回收,此时幸存者0区还幸存的对象会放在幸存者1区,下次再进行垃圾回收会回收对伊甸园区和幸存者1区的对象,把还幸存的对象放到幸存者0区,循环往复(谁空谁是to)
默认15次,即经过Minor GC15次的对象会进入老年代,可以通过“-XX:MaxTenuringThreshold=threshold,由于Java对象头储存年龄只有4个bit位,所以MaxTenuringThreshold的值最大只能是15.

总结:
JVM在进行GC的时候,并不是每次都对新生代、老年代、方法区一起回收的,大部分回收的都是新生代。
Minor GC或Young GC,新生代(Eden、S0、S1)收集,当年轻代空间不足时,就会触发Young GC,这里指的是Eden区满,S1,S0满不会触发GC。Java的大多数对象都是朝生夕死的,所以Young GC非常频繁,一般回收速度也比较快。Young GC时会引发STW,暂停其他用户线程,等垃圾回收结束,用户线程才恢复运行。
Major GC或Old GC 老年代的垃圾收集
整堆回收,Full GC,对整个java堆和方法区的垃圾收集。
触发Full GC的情况有五种
1.调用System.gc()时,系统建议执行Full GC,但不是必然执行的。
2.老年代空间不足
3.方法区空间不足
4.通过Minor GC后进入老年代的平局大小大于老年代的空闲内存。
5.由Eden区、s0区向s1区复制时,对象大小大于s1区的可用内存,则把该对象转移到老年代,并且老年代的可用内存小于该对象的大小
其实不分代完全可以,分代的唯一理由就是优化GC性能。如果没有分代,那所有的对象都在一块,就如同把一个学校的人都关在一个教室。GC的时候要找到哪些对象没用,这样就会对堆的所有区域进行扫描。而很多对象都是朝生夕死的,如果分代的话,把新创建的对象放到某一地方,当GC的时候先把这块存储“朝生夕死”对象的区域进行回收,这样就会腾出很大的空间出来。
为什么要有TLAB(Thread Local Allocation Buffer)?
什么是TLAB?
额外说明

-XX:+PrintFlagsInitial 查看所有参数的默认初始值
-XX:+PrintFlagsFinal 查看所有参数最终修改过的值
具体查看某个参数的指令(在cmd或linux终端中)
①jps 查看程序的具体进程
②jinfo -flag SurvivorRatio(参数名称) 78856(进程id)
-Xms 初始堆空间内存(默认为物理内存的1/64)
-Xmx 最大堆空间内存(默认为物理内存的1/4)
-Xmn 设置新生代的大小(初始大小,也是最大大小)
-XX:NewRatio:配置新生代和老年代的结构占比
-XX:SurvivorRatio 设置新生代中Eden和S区的空间占比
-XX:MaxTenuringThreshold 设置新生代S区对象最大年龄
-XX:PrintGCDetails 输出详细的GC处理日志
打印GC的简要信息:① -XX:PrintGC ②-verbose:gc
-XX:HandlePromotionFailure 是否设置分配担保
在发生Minor GC之 前,虚拟机会检查老年代最大可用的连续空间是否大于新生代所有对象的总空间。
如果大于,则此次Minor GC是安全的
如果小于,则虚拟机会查看-XX: HandlePromotionFailure设置值是否允许担保失败。
➢如果HandlePromotionFailure=true, 那么会继续检查老年代最大可用连续空间是否大于历次晋升到老年代的对象的平均大小。
在JDK6 Update24之后(JDK7) ,HandlePromotionFailure参 数不会再影响到虛拟机的空间分配担保策略,观察OpenJDK中的源码变化,虽然源码中还定义了HandlePromotionFailure参数,但是在代码中已经不会再使用它。JDK6 Update24(JDK7)之后的规则变为只要老年代的连续空间大于新生代对象总大小或者历次晋升的平均大小就会进行Minor GC, 否则将进行Full GC。