废话不多说,首先发一段代码:
public class ExplicitStatic2 {
public static void main(String[] args) {
}
static Eat eat = new Eat();
/*static{
System.out.println();
}
static Eat eat2 = new Eat();*/
}
class Eat{
static Eat Eat,Eat1;
Cup cup1 = new Cup(1);
static Cup cup2;
static {
Eat1 = new Eat();
System.out.println();
cup2 = new Cup(2);
System.out.println();
Eat = new Eat();
System.out.println();
}
Eat(){
System.out.println("Eat()");
}
Cup cup3 = new Cup(3);
}
class Cup {
Cup(int marker) {
System.out.println("Cup(" + marker + ")");
}
void f(int marker) {
System.out.println("f(" + marker + ")");
}
}
运行结果:
Cup(1)
Cup(3)
Eat()
Cup(2)
Cup(1)
Cup(3)
Eat()
Cup(1)
Cup(3)
Eat()
看到运行结果,相信有些人已经迷惑了,实际上我也迷惑了好久。这里仅仅说说我的看法,并没有找到相关证明。
从主类中可以看到,main()方法中并没有写任何代码,但是,在主类中有一个静态域,就是变量eat。显然静态代码是在类加载时自动执行的,那么,程序的入口便是这个变量eat,那么我们就对它进行始使化。遵循着静态代码先初始化------非静态代码初始化----最后构造函数这个规则,我们一步一步往下运行。首先程序会加载Eat这个类,并对该类中的静态代码初始化,也就是static块会被先初始化。进入static块中,里面有一个eat1 = new Eat(),显然,这个静态变量也要进行初始化,那么,这个时候程序就会从Eat这个类中重新进入,就会又进入到这个块中,但不会再执行创建eat1,因为它是静态的,只占一份内存空间。那为什么程序没有进入System.out.println()这句话呢?这是显而易见的,因为eat1的确还没有创建完成,它还没有对非静态域进行操作,也没执行构造方法,所以,在没完成这两项任务之前,它是不会执行System.out.println()这个的。那么,我们就会想到,不管静态初始化过程中发生了什么,后续过程中一定会打印出Cup(1) Cup(3) Eat(),这才代表eat1这个静态变量创建完成。然后,才会找印一行空格,再执行下一个静态变量cup2的创建。后面的过程都基本一样,这里就不详述了!