问题
public class Eat {
public static final Eat INSTANCE=new Eat();
private final int beltSize;
private static final int CURRENT_YEAR= Calendar.getInstance().get(Calendar.YEAR);
private Eat(){
System.out.println("constructor");
beltSize=CURRENT_YEAR-1930;
}
public int beltSize(){
System.out.println("getBeltSize");
return beltSize;
}
public static void main(String[] args){
System.out.println("I wear a size "+INSTANCE.beltSize()+" belt");
}
}
这里的结果并不是2017-1930,而是0-1930
解释:main方法的调用导致类开始初始化,静态域被赋值为缺省值,也就是INSTANCE为null,CURRENT_YEAR为0。接下来,静态域初始器按照其出现的顺序执行,也就是闲执行new Eat();此时beatSize被赋值为0-1930,即CURRENT_YEAR还没来得及初始化就已经被使用了。也就是造成结果的原因。
注意
在final类型的静态域在被初始化之前,存在读取其值的可能性,而此时该静态域包含的还只是其所属类型的缺省值。与直觉相违背的是,我们通常会将final类型的域看做常量。final类型的域只有在其初始化表达式是常量表达式时才是常量。
类的初始化
这里引出了一个概念,类的初始化。
类初始化过程
包含类静态初始化器和静态域的初始化器
类初始化发生的条件
- 创建T类的实例
- T类或者T接口静态方法被调用
- T类或者T接口被赋值
- T类或者T接口声明的静态域被调用(非常量)
- 一个类初始化,其父类也会被初始化(初始化一个接口则不会)
- 对静态域的引用会导致定义这个域的类或者接口初始化