JVM的动态内存分配和回收,已经相当成熟。本篇文章主要讨论JVM对于堆和方法区的垃圾回收问题。
什么样的对象需要被回收?
已死的对象。(不可能再被任何途径使用的对象)
如何判断对象已死?
在对象中添加一个引用计数器,每当有一个地方引用它时,计数器值就加一; 当引用失效时,计数器值就减一; 任何时刻计数器为零的对象就是不可能再被使用的对象。
引用计数算法(Reference Counting)虽然占用了一些额外的内存空间来进行计数,但它的原理简单,判定效率也很高,在大多数情况下它都是一个不错的算法。但是,在主流的Java虚拟机里面都没有选用引用计数算法来管理内存,主要原因是,这个看似简单 的算法有很多例外情况要考虑,必须要配合大量额外处理才能保证正确地工作。譬如单纯的引用计数就很难解决对象之间相互循环引用的问题。
举个例子:下面的代码中创建了A和B两个对象,并互相引用。在后面又把A和B对象置为null,此时A和B对象应该是属于用不到的对象了,但由于引用关系还互相存在,导致这两个对象没有办法被gc处理掉,从而导致内存泄露。
public class RefereceCount {
public Object instance = null;
// 占用内存,方便gc日志中查看此对象是否被回收
private Byte[] bys = new Byte[1024 * 1024 * 2];
public static void main(String[] args) {
RefereceCount rcA = new RefereceCount();
RefereceCount rcB = new RefereceCount();
rcA.instance = rcB;
rcB.instance =