1 如何判断对象可以回收
1.1 引用计数法
- 循环引用问题: 引用计数法无法处理循环引用的情况。如果两个或多个对象之间相互引用,并且彼此之间的引用计数都不为零,那么这些对象将永远无法被释放,即使它们不再被程序使用。
1.2 可达性分析法
可达性分析法的基本思想是从一组称为“GC Roots”的根对象出发,通过遍历对象之间的引用关系,找到所有与根对象直接或间接相连的对象,这些对象被认为是可达的(活动的),而那些不与根对象相连的对象则被认为是不可达的(垃圾对象)。
在 JVM 中,GC Roots 是一组特殊的对象,包括:
- 虚拟机栈(Java 虚拟机栈中引用的对象)
- 本地方法栈(本地方法引用的对象)
- 方法区中静态属性引用的对象
- 方法区中常量引用的对象
虚拟机栈中引用的对象: 虚拟机栈中存储着每个线程的方法调用栈,每个栈帧中都包含了局部变量表,其中存储了对对象的引用。这些引用是方法调用过程中传递的参数、局部变量以及方法返回值。例如:
public class GCRootsExample {
public static void main(String[] args) {
Object obj = new Object(); // 虚拟机栈中的局部变量引用了一个对象
// ...
}
}
方法区中静态属性引用的对象: 方法区中存储着类的静态变量和常量,静态变量中存储的对象引用也可以作为 GC Roots。例如:
public class GCRootsExample {
private static Object staticObj = new Object(); // 静态属性引用了一个对象
// ...
}
方法区中常量引用的对象: 方法区中的常量池存储着字符串常量、类常量等,其中引用的对象也可以作为 GC Roots。例如:
public class GCRootsExample {
private final static String constantString = "constant"; // 常量引用了一个字符串对象
// ...
}
本地方法栈中引用的对象: 本地方法栈用于执行本地方法,其中可能会有对对象的引用。例如:
public class GCRootsExample {
public native void nativeMethod(); // 本地方法声明
// ...
}
JNI 中引用的对象: Java Native Interface(JNI)允许 Java 程序与本地代码(如 C、C++)交互,在 JNI 中引用的对象也可以作为 GC Roots。例如:
JNIEXPORT void JNICALL Java_GCRootsExample_nativeMethod(JNIEnv *env, jobject obj) {
jobject localRef = (*env)->NewObject(env, jclass, methodID); // JNI 中的本地对象引用
// ...
}
1 系统对象
2 本地方法
3 被加锁对象
1.3 四种引用
- 强引用(Strong Reference): 强引用是 Java 中最常见的引用类型。当一个对象被强引用持有时,即使内存不足,垃圾回收器也不会回收这个对象。只有当所有强引用都被释放时,对象才会被垃圾回收器回收。
- 软引用(Soft Reference): 软引用是一种比强引用弱的引用类型。当内存不足时,垃圾回收器可能会回收被软引用持有的对象。软引用通常用于实现内存敏感的缓存系统,使得在内存不足时可以回收缓存中的对象,从而避免内存溢出。
- 弱引用(Weak Reference): 弱引用是比软引用更弱的引用类型。当一个对象只被弱引用持有时,即使内存充足,垃圾回收器也会回收这个对象。弱引用通常用于实现对象的辅助数据结构,如 ThreadLocalMap 中对 ThreadLocal 对象的引用。
- 虚引用(Phantom Reference): 虚引用是最弱的引用类型。虚引用的存在不会影响对象的生命周期,即使对象只被虚引用持有时,垃圾回收器也会将其回收。虚引用通常用于跟踪对象被回收的时机,可以在对象被回收前执行一些特定的操作。
2 分代垃圾回收
- 对象首先分配在伊甸园区域
- 新生代空间不足时,触发minor ge,伊甸园和from存活的对象使用copy复制到to中,存活的对象年龄加1并且交换from to
- minor ge会引发stop the world,.暂停其它用户的线程,等垃圾回收结束,用户线程才恢复运行
- 当对象寿命超过阀值时,会晋升至老年代,最大寿命是 15 - 4bit
- 当老年代空间不足 ,会先尝试触发 minor gc 如果之后空间任然不足,那么触发 full gc stop the world 的时间更长
2.1 相关jvm 参数
2.2 大对象
如果一个对象的内存超过了新生代的容量,以至于无法在新生代中分配足够的空间来存储该对象,那么这个对象就会被直接分配到老年代中。这种情况通常会发生在对象比较大或者生存时间比较长的情况下。