赞
踩
Java 是一门不需要自己手动控制内存释放的语言,有自动的垃圾回收机制,也就是我们熟悉的GC(Garbage Collection)。有了垃圾回收机制后,程序员只需要关心内存的申请即可,内存的释放由系统自动识别完成。
在进行垃圾回收时,不同的对象引用类型,GC会采用不同的回收时机,这时自动的垃圾回收的算法就会变得非常重要了,如果因为算法的不合理,导致内存资源一直没有释放,同样也可能会导致内存溢出的。
主要思想是通过记录对象被引用的次数来判断对象是否可回收。当一个对象被创建时,引用计数加一;当一个引用指向该对象时,引用计数也加一;当引用失效或对象被销毁时,引用计数减一。当对象的引用计数变为零时,即表示该对象不再被任何引用指向,可以被垃圾收集器回收。
String demo = new String("123");
String demo = null;
JVM已经放弃了引用计数算法,这是因为当对象间出现了循环引用的话,则引用计数法就会失效。
虽然a和b都为null,但是由于a和b存在循环引用,这样a和b永远都不会被回收。
该算法会存在一个根对象【GC Roots】,递归地遍历所有可通过引用链访问到的对象,搜索过的路径被称为(Reference Chain),判断某对象是否与根对象有直接或间接的引用,如果没有被引用,则可以当做垃圾回收。现在JVM采用的都是通过可达性分析算法来确定哪些内容是垃圾。
X,Y这两个节点是可回收的,但是并不会马上的被回收。对象中存在一个方法finalize
。当对象被标记为可回收后,当发生GC时,首先会判断这个对象是否执行了finalize
方法,如果这个方法还没有被执行的话,那么就会先来执行这个方法,接着在这个方法执行中,可以设置当前这个对象与GC ROOTS产生关联,那么这个方法执行完成之后,GC会再次判断对象是否可达,如果仍然不可达,则会进行回收,如果可达了,则不会进行回收。
finalize
方法对于每一个对象来说,只会执行一次。如果第一次执行这个方法的时候,设置了当前对象与RC ROOTS关联,那么这一次不会进行回收。 那么等到这个对象第二次被标记为可回收时,那么该对象的finalize方法就不会再次执行了。
根对象是那些肯定不能当做垃圾回收的对象,就可以当做根对象。在Java中,GC Roots
包括四种:
/**
* demo是栈帧中的本地变量,当demo = null时,由于此时demo充当了GC Root的作用,
* demo与原来指向的实例new Demo()断开了连接,对象被回收。
*/
public class Demo {
public static void main(String[] args) {
Demo demo = new Demo();
demo = null;
}
}
/**
* 当栈帧中的本地变量b = null时,由于b原来指向的对象与GC Root (变量b) 断开了连接,
* 所以b原来指向的对象会被回收,而由于我们给a赋值了变量的引用,a在此时是类静态属性引用,
* 充当了 GC Root 的作用,它指向的对象依然存活!
*/
public class Demo {
public static Demo a;
public static void main(String[] args) {
Demo b = new Demo();
b.a = new Demo();
b = null;
}
}
/**
* 常量 a 指向的对象并不会因为 demo 指向的对象被回收而回收
*/
public class Demo {
public static final Demo a = new Demo();
public static void main(String[] args) {
Demo demo = new Demo();
demo = null;
}
}
任何 Native 接口都会使用某种本地方法栈,实现的本地方法是使用 C 连接模型的话,那么它的本地方法栈就是 C 栈。
当线程调用 Java 方法时,虚拟机会创建一个新的栈帧并压入 Java 栈,然而当它调用的是本地方法时,虚拟机会保持 Java 栈不变,虚拟机只是简单地动态连接并直接调用指定的本地方法。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。