最近在补《thinking in java》...有一节提到了加载类需要做的一些准备...我照着书本敲了一下代码...同时稍微修改了一下书本上的代码....
1 packagecharpter14;2
3 importjava.util.Random;4
5
6 classInitable{7 static{8 System.out.println("haha");9 }10 static final int staticFinal = 47;11 static final int staticFinal2 = ClassInitialization.rand.nextInt(1000);12 static{13 System.out.println("Initialing Initable");14 }15 }16
17 classInitable2{18 static int staticNonFinal = 147;19 static{20 System.out.println("Initialing Initable2");21 a = 1;22 }23 static int a = 7;24 }25
26 classInitable3{27 static int staticNonFinal = 74;28 static{29 System.out.println("Initialing Initable3");30 }31 }32
33 public classClassInitialization {34 public static Random rand = new Random(47);35 public static void main(String[] args) throwsException {36 Class initable = Initable.class;37 System.out.println("After creating Initable ref");38 System.out.println(Initable.staticFinal);39 System.out.println(Initable.staticFinal2);40 System.out.println(Initable2.staticNonFinal);41 Class initable3 = Class.forName("charpter14.Initable3");42 System.out.println("After creating Initable3 ref");43 System.out.println(Initable3.staticNonFinal);44 System.out.println(Initable2.a);45 }46 }
输出结果:
After creating Initable ref
47
haha
Initialing Initable
258
Initialing Initable2
147
Initialing Initable3
After creating Initable3 ref
74
7
分析
Class initable = Initable.class;的时候什么都没有输出,说明了调用.class的时候并不会初始化Initable类.
47 haha Initialing Initable 258; 调用Initable.staticFinal的时候因为staticFinal变量是static final 并且是 基本数据类型字面值,不是通过方法得到的值的时候也不会初始化这个类. 调用staticFinal2的时候,因为调用了其他类的方法,所以要先初始化Initable这个类,所以会去做static区块的内容.因此会输出haha 和 Initialing Initable,然后初始化完毕,再输出258.
调用Initable2.staticNonFinal的时候,因为这个变量不是final类型的,所以要先初始化类,所以去做static区块,因此输出Initialing Initable2以后再输出47.
问题
最后我想一个问题,这点有点奇怪....我有点想不明白...输出Initable2.a的值是7不是1.说明了Initable2的static区块中的a=1是先于赋值static int a = 7执行的...这里有个问题就是..既然如此的话那执行a=1的时候a变量的类型都是不确定的为什么能赋值呢...我想可能是因为static int a = 7的时候..第一步做的是static int a=0,就是先分配空间,没有赋值; 然后执行static区块,赋值a = 1,这样a的类型就确定了,能赋值.然后再执行a = 7.如果确实是这样的话..那还是有一个问题.....(Σ( ° △ °|||)︴问题好多啊).....
public classTest {static{
a= 99;
System.out.println(a);
}static int a = 11;
}
如果刚才的推论是正确的话这里应该会输出99....因为先给a分配空间,再赋值a=99;再输出a,就是99...然后再赋值a=11.....
可是实际上eclipse会提示你这里有编译错误..
eclipse会告诉你Cannot reference a field before it is defined.
它会告诉你a是没有定义的,所以你这里不能输出a....
这点我一直想不明白...