首先看下下面的程序打印什么?
public class Final_init {
public static final Final_init INSTANCE = new Final_init();
private final int beltSize;
private static final int CURRENT_YEAR=
Calendar.getInstance().get(Calendar.YEAR);
private Final_init(){
beltSize = CURRENT_YEAR - 1930;
}
public int beltSize(){
return beltSize;
}
public static void main(String[] args) {
System.out.println(INSTANCE.beltSize());
}
}
看起来是当前年份减1930,实际你发现它打印-1930
该问题是由 类初始化顺序中的循环引起的
1.Final_init类得初始化由main方法调用触发
2.首先静态域设置为缺省值
3.接着静态初始域顺序执行,第一个INSTANCE 是通过调用Final_init()构造器计算而来
4.构造器设计静态域CURRENT_YEAR的表达式来初始化beltSize。
通常读取一个静态域是引起一个类被初始化的事件之一,但我们刚已经初始化Final_init了,递归初始化会被忽略掉。
因此,CURRENT_YEAR的值仍旧是缺省值0。导致beltSize为-1930。
5.等构造器返回后,CURRENT_YEAR被初始化成了当前年份。可是太晚了 - @ -
所有要当心初始化循环。
还有 final类型的静态变量被初始化之前,存在着读取其值的可能。final类型的域只有在初始化表达式是常量表达式时才是常量。