Java类中各种静态变量的加载顺序的学习

最近在补《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

 


分析

  1. Class initable = Initable.class;的时候什么都没有输出,说明了调用.class的时候并不会初始化Initable类.
  2. 47 haha Initialing Initable 258; 调用Initable.staticFinal的时候因为staticFinal变量是static final 并且是 基本数据类型字面值,不是通过方法得到的值的时候也不会初始化这个类. 调用staticFinal2的时候,因为调用了其他类的方法,所以要先初始化Initable这个类,所以会去做static区块的内容.因此会输出haha 和 Initialing Initable,然后初始化完毕,再输出258.

  3. 调用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....

这点我一直想不明白...

 

posted @ 2015-06-10 21:35 abcwt112 阅读(...) 评论(...) 编辑 收藏