1、YGC做了哪些操作
Eden区:
1)对象回收
2)存活对象 被复制到 Survivor区的"To"(age不加1)
Survivor的From:
(3)对象回收
存活对象
4)不满足年龄条件的被复制到 Survivor区的“To”(age加1)
5)满足年龄条件的移入年老代
2、GC中的一些点
1、Minor GC会一直重复这样的过程,直到"To"区被填满,会将所有对象移动到年老代中。
2、动态判断对象的年龄。如果Survivor区中相同年龄的所有对象大小的总和大于Survivor空间的一半,
年龄大于或等于该年龄的对象可以直接进入老年代。
3、空间分配担保。每次进行Minor GC时,JVM会计算Survivor区移至老年区的对象的平均大小,
如果这个值大于老年区的剩余值大小则进行一次Full GC,如果小于检查HandlePromotionFailure设置,
如果true则只进行Monitor GC,如果false则进行Full GC。
4、当使用G1,CMS 时,FullGC发生的时候 是 Serial+SerialOld。
当使用ParalOld时,FullGC发生的时候是 ParallNew +ParallOld.
3、JVM参数
-XX:NewSize 设置年轻代的初始大小
-XX:MaxNewSise 设置年轻代的最大值
建议设为整个堆大小的1/3或者1/4,两个值设为一样大。
-XX:SurvivorRatio
设置Eden和其中一个Survivor的比值,默认是8比1。
-Xmx 堆的最大值;
-Xms 堆的最小值;
两个值最好设置为相等。
-XX:+PrintTenuringDistribution
用于显示每次Minor GC时Survivor区中各个年龄段的对象的大小。
-XX:MaxTenuringThreshold和-XX:InitialTenuringThreshold
用于设置Survivor区对象晋升年老代中年龄的最小值和最大值
-XX:NewRatio=1
修改新生代和老年代之间的内存比例;
4、1.6、1.7、1.8关于永久代的变化
移除永久代的工作从JDK1.7就开始了。JDK1.7中,存储在永久代的部分数据就已经转移到了Java Heap或者是 Native Heap。
但永久代仍存在于JDK1.7中,并没完全移除,譬如符号引用(Symbols)转移到了native heap;
字面量(interned strings)转移到了java heap;类的静态变量(class statics)转移到了java heap。
static String base = "string";
public static void main(String[] args) {
List list = new ArrayList();
for (int i=0;i< Integer.MAX_VALUE;i++){
String str = base + base;
base = str;
list.add(str.intern());
}
}
JDK 1.6 的运行结果:PG SPACE
JDK 1.7 的运行结果:java.lang.OutOfMemoryError: Java heap space
JDK 1.8 的运行结果:java.lang.OutOfMemoryError: Java heap space
可以大致验证 JDK 1.7 和 1.8 将字符串常量由永久代转移到堆中。
5、Metaspace(元空间)
无