导入与编译代码中的任何指令不相关。他们只在编译时建立别名。
有一些反映方法允许类加载但尚未初始化,但在大多数情况下,您可以假设每当引用类时,它都已被初始化。
执行静态成员初始化器和静态块,就像它们是源代码顺序中所有的一个静态初始化程序块一样。
通过静态成员变量引用的对象将被强制引用,直到该类被卸载为止。一个普通的ClassLoader永远不会卸载一个类,但应用程序服务器使用的类在正确的条件下执行。然而,这是一个棘手的领域,并且是许多难以诊断的内存泄漏的源泉 – 另一个原因是不使用全局变量。
作为(切向)奖金,这里有一个棘手的问题要考虑:
public class Foo {
private static Foo instance = new Foo();
private static final int DELTA = 6;
private static int BASE = 7;
private int x;
private Foo() {
x = BASE + DELTA;
}
public static void main(String... argv) {
System.out.println(Foo.instance.x);
}
}
这段代码打印?尝试一下,你会看到它打印“6”。这里有一些工作,一个是静态初始化的顺序。代码被执行,好像它是这样写的:
public class Foo {
private static Foo instance;
private static final int DELTA = 6;
private static int BASE;
static {
instance = null;
BASE = 0;
instance = new Foo(); /* BASE is 0 when instance.x is computed. */
BASE = 7;
}
private int x;
private Foo() {
x = BASE + 6; /* "6" is inlined, because it's a constant. */
}
}