本文总结自:《深入理解Java虚拟机》,本文用于帮助记忆,如果想要详细准确的描述,请阅读此书
如何判断对象已死:
1. 引用计数器法:当一个地方引用它时,就+1;当引用失效时,就-1;GC会回收那些计数为0的对象。
优点:简单,快捷
缺点:对于相互引用的对象会失效
2. 跟搜索算法:从一系列名为“GC Roots”的对象为起始点向下搜索,搜索经过的路径就是引用链;当对象位于引用链上时,就不回收,当对象间虽然相连,但是却不在引用链上时就是回收这一部分。如下图,Object 1,Object2,Object3 位于引用链上,就不回收;Object 4、Object 5、Object 6虽然相连,但是不位于引用链上,就会被回收。
可以作为GC Roots的对象为:
(1). 方法区常量、类静态静态属性所引用的类
(2). 虚拟机栈(栈帧中的本地变量表)中的引用的对象
(3). 本地方法栈中JNI(即一般说的native方法)中引用的对象
3. 引用的分类:强引用、软引用、弱引用、虚引用
强引用:如Object a=new Object()只要强引用在,垃圾回收器就不会回收掉该对象
软引用:是一些有用但是不是必须的对象,一般情况下不会回收,在将要发生内存溢出时,才会回收掉
弱引用:在第一轮的垃圾回收中,便会被回收掉
虚引用:最弱的一中引用,对对象生存时间,其作用实在被垃圾回收时,收到一个系统通知(我也不知道这个系统通知的作用)
4. 垃圾回收会有两次标记过程
当发现对象不在引用链上时,将会发生一次标记,如果该对象没有必要执行finalize()方法时即:该对象没有finalize()方法或者finalize()方法已经被调用过,则进行回收。
当发现有finalize()方法时,则会将该对象放在一个专有的finalize线程中去执行。该线程会触发此方法,但是不保证执行成功。如果在finalize()方法中,该对象重新被引用,则逃过一劫,否则将会被回收掉
注意:finalize()方法时java向c++程序员妥协的产物,不建议使用。