类是在任何static成员被访问时加载的(构造器也是static方法)。类的整个加载过程包括加载、验证、准备、解析、初始化5个阶段。我这里只讨论我们在笔试题中比较关心的、影响程序输出的部分。
类加载:
在准备阶段,static变量在方法区被分配内存,然后内存被初始化零值(注意和static变量初始化的区别)。
在初始化阶段,执行类构造器<clinit>()方法(注意和实例构造器<init>()方法不同)。虚拟机会保证子类的<clinit>()方法执行之前,父类的<clinit>()方法已经执行。
在执行<clinit>()方法时,按照类定义中static变量的赋值语句和static代码段的书写顺序,依次执行。
子类调用基类的静态方法时,相当于基类调用自己的静态方法,所以子类的static不会初始化。例子如下:
Child.sMethodBase(); // 类的定义在最后面
这一句的执行结果为:
基类initPrint2 静态变量s4:null
基类静态方法sMethodBase 静态变量s4:基类静态变量s4
创建对象:
虚拟机在遇到new指令时,首先检查类是否加载过,在类加载检查通过后,虚拟机为对象分配内存,分配完内存后会将内存空间初始化为零值(不包括对象头)。所以对象的实例字段在初始化之前就有了零值。
执行new指令之后会接着执行实例构造器<init>方法,这时才开始对象的初始化。
进入构造器时,如果有基类,会进入基类的无参构造器(或者用super()显式指定的基类构造器)。在构造之前,先按照实例字段和非static代码段的书写顺序,依次初始化,最后执行构造器的语句。
super()语句要按基类的次序,放