最近在补《thinking in java》...有一节提到了加载类需要做的一些准备...我照着书本敲了一下代码...同时稍微修改了一下书本上的代码....
1 package charpter14; 2 3 import java.util.Random; 4 5 6 class Initable{ 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 class Initable2{ 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 class Initable3{ 27 static int staticNonFinal = 74; 28 static{ 29 System.out.println("Initialing Initable3"); 30 } 31 } 32 33 public class ClassInitialization { 34 public static Random rand = new Random(47); 35 public static void main(String[] args) throws Exception { 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 class Test { 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....
这点我一直想不明白...