jvm -垃圾回收

1 如何判断对象可以回收

1.1 引用计数法

  • 循环引用问题: 引用计数法无法处理循环引用的情况。如果两个或多个对象之间相互引用,并且彼此之间的引用计数都不为零,那么这些对象将永远无法被释放,即使它们不再被程序使用。

在这里插入图片描述

1.2 可达性分析法

可达性分析法的基本思想是从一组称为“GC Roots”的根对象出发,通过遍历对象之间的引用关系,找到所有与根对象直接或间接相连的对象,这些对象被认为是可达的(活动的),而那些不与根对象相连的对象则被认为是不可达的(垃圾对象)。

在 JVM 中,GC Roots 是一组特殊的对象,包括:

  1. 虚拟机栈(Java 虚拟机栈中引用的对象)
  2. 本地方法栈(本地方法引用的对象)
  3. 方法区中静态属性引用的对象
  4. 方法区中常量引用的对象

虚拟机栈中引用的对象: 虚拟机栈中存储着每个线程的方法调用栈,每个栈帧中都包含了局部变量表,其中存储了对对象的引用。这些引用是方法调用过程中传递的参数、局部变量以及方法返回值。例如:

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 四种引用

  1. 强引用(Strong Reference): 强引用是 Java 中最常见的引用类型。当一个对象被强引用持有时,即使内存不足,垃圾回收器也不会回收这个对象。只有当所有强引用都被释放时,对象才会被垃圾回收器回收。
  2. 软引用(Soft Reference): 软引用是一种比强引用弱的引用类型。当内存不足时,垃圾回收器可能会回收被软引用持有的对象。软引用通常用于实现内存敏感的缓存系统,使得在内存不足时可以回收缓存中的对象,从而避免内存溢出。
  3. 弱引用(Weak Reference): 弱引用是比软引用更弱的引用类型。当一个对象只被弱引用持有时,即使内存充足,垃圾回收器也会回收这个对象。弱引用通常用于实现对象的辅助数据结构,如 ThreadLocalMap 中对 ThreadLocal 对象的引用。
  4. 虚引用(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 大对象

如果一个对象的内存超过了新生代的容量,以至于无法在新生代中分配足够的空间来存储该对象,那么这个对象就会被直接分配到老年代中。这种情况通常会发生在对象比较大或者生存时间比较长的情况下。

  • 31
    点赞
  • 46
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值