java解惑49--类初始化

问题

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接口声明的静态域被调用(非常量)
  • 一个类初始化,其父类也会被初始化(初始化一个接口则不会)
  • 对静态域的引用会导致定义这个域的类或者接口初始化
java解惑49--类初始化_第1张图片
类初始化

你可能感兴趣的:(java解惑49--类初始化)