匿名类是否*始终*维护对其封闭实例的引用?

我正在处理一些代码,其中一个对象“foo”正在创建另一个对象“bar”,并将其传递给.在此之后,foo将返回bar,然后我希望foo变得无法访问(即:可用于垃圾回收)。Callable

我最初的想法是匿名创建。例如:Callable

class Foo {
  ...

  public Bar createBar() {
    final int arg1 = ...
    final int arg2 = ...
    final int arg3 = ...
    return new Callable<Baz>() {
      @Override
      public Baz call() {
        return new Baz(arg1, arg2, arg3);
      }
    };
  }
}

然而,我突然想到,这可能实际上并不像预期的那样工作,因为内部类通常会保留对其封闭对象的引用。我不想在这里引用封闭类,因为我希望在 仍然可访问时收集封闭对象。Callable

另一方面,检测封闭实例从未被实际引用应该是相当微不足道的,所以也许Java编译器足够聪明,在这种情况下不包括引用。

所以。。。匿名内部类的实例是否会保留对其封闭实例的引用,即使它从未实际使用封闭实例引用?


答案 1

是的,匿名内部类的实例保留对其封闭实例的引用,即使这些引用从未实际使用过。此代码:

public class Outer {
  public Runnable getRunnable() {
    return new Runnable() {
      public void run() {
        System.out.println("hello");
      }
    };
  }
}

编译时生成两个类文件,和 。反汇编后者,匿名内部类,得到:javacOuter.classOuter$1.classjavap -c

Compiled from "Outer.java"
class Outer$1 extends java.lang.Object implements java.lang.Runnable{
final Outer this$0;

Outer$1(Outer);
  Code:
   0:   aload_0
   1:   aload_1
   2:   putfield        #1; //Field this$0:LOuter;
   5:   aload_0
   6:   invokespecial   #2; //Method java/lang/Object."<init>":()V
   9:   return

public void run();
  Code:
   0:   getstatic       #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   3:   ldc     #4; //String hello
   5:   invokevirtual   #5; //Method java/io/PrintStream.println:(Ljava/lang/String;)V
   8:   return

}

该行显示构造函数正在将对封闭实例的引用存储在字段(类型)中,即使此字段不再使用。putfieldthis$0Outer

如果您尝试使用匿名内部类创建可能长期存在的小型对象,这是不幸的,因为它们将保留(可能很大的)封闭实例。解决方法是改用静态类(或顶级类)的实例。不幸的是,这更加冗长。


答案 2

通过在类中引入静态方法,可以轻松地将嵌套的匿名类转换为“静态”匿名类。

import java.util.ArrayList;


public class TestGC {
    public char[] mem = new char[5000000];
    public String str = "toto";

    public interface Node {
        public void print();
    }

    public Node createNestedNode() {
        final String str = this.str;
        return new Node() {
            public void print() {
                System.out.println(str);
            }
        };
    }

    public static Node createStaticNode(TestGC test) {
        final String str = test.str;
        return new Node() {
            public void print() {
                System.out.println(str);
            }
        };
    }

    public Node createStaticNode() {
        return createStaticNode(this);
    }

    public static void main(String... args) throws InterruptedException {
        ArrayList<Node> nodes = new ArrayList<Node>();
        for (int i=0; i<10; i++) {
            // Try once with createNestedNode(), then createStaticNode()
            nodes.add(new TestGC().createStaticNode());
            System.gc();
            //Thread.sleep(200);
            System.out.printf("Total mem: %d  Free mem: %d\n", Runtime.getRuntime().totalMemory(), Runtime.getRuntime().freeMemory());
        }
        for (Node node : nodes)
            node.print();
        nodes = null;
        System.gc();
        //Thread.sleep(200);
        System.out.printf("Total mem: %d  Free mem: %d\n", Runtime.getRuntime().totalMemory(), Runtime.getRuntime().freeMemory());
    }
}