一、GC触发时机
- 会在cpu空闲的时候自动进行回收
- 在堆内存存储满了之后
- 主动调用System.gc()后尝试进行回收
补充:System.gc()用于调用垃圾收集器,在调用时,垃圾收集器将运行以回收未使用的内存空间。它将尝试释放被丢弃对象占用的内存。 然而System.gc()调用附带一个免责声明,无法保证对垃圾收集器的调用。 所以System.gc()并不能说是完美主动进了垃圾回收。
二、哪些垃圾需要回收
有三个区是不需要进行垃圾回收的,分别是:程序计数器、JVM栈、本地方法栈。也就是线程私有区不进行回收。
需要进行回收垃圾的区:堆和方法区。也就是线程共享区需要回收。
三、什么样的对象才是垃圾
对于Java对象来讲,如果说这个对象没有被其他对象所引用该对象就是无用的,此对象就被称为垃圾,其占用的内存也就要被销毁。
- 该类的所有实例对象都已经被回收。
- 加载该类的ClassLoader已经被回收。
- 该类对应的反射类java.lang.Class对象没有被任何地方引用。
四、垃圾标记算法
Java中标记垃圾的算法主要有两种, 引用计数法和可达性分析算法。
1、引用计数法
简单的来说就是判断对象的引用数量。实现方式:给对象共添加一个引用计数器,每当有引用对他进行引用时,计数器的值就加1,当引用失效,也就是不在执行此对象,它的计数器的值随之减1,若某一个对象的计数器的值为0,那么表示这个对象没有被其他对象引用,也就是意味着是一个失效的垃圾对象,就会被gc进行回收。 但是这种简单的算法在当前的jvm中并没有采用,原因是他并不能解决对象之间循环引用的问题。 假设有A和B两个对象之间互相引用,也就是说A对象中的一个属性是B,B中的一个属性时A,这种情况下由于他们的相互引用,从而垃圾回收机制无法识别。
- 优点: 执行效率高,程序执行受影响较小
- 缺点:无法检测出循环引用的情况,导致内存泄露