03-再谈初始化顺序
在01节做了小结对象的创建过程:
请考虑一个名为Dog 的类:
(1) 类型为Dog 的一个对象首次创建时,或者Dog 类的static 方法/static 字段首次访问时,Java 解释器必须找到Dog.class(在事先设好的类路径里搜索)。
(2) 找到Dog.class 后(它会创建一个Class 对象),它的所有static 初始化模块都会运行。因此,static 初始化仅发生一次——在Class 对象首次载入的时候。
(3) 创建一个new Dog()时,Dog 对象的构建进程首先会在内存堆(Heap)里为一个Dog 对象分配足够多的存储空间。
(4) 这种存储空间会清为零,将Dog 中的所有基本类型设为它们的默认值(零用于数字,以及boolean 和char 的等价设定)。
(5) 进行属性定义时发生的所有初始化都会执行(上面的第一个例子)。
(6) 执行构造方法。
那么看下面一个例子:
//: PolyConstructors.java // Constructors and polymorphism // don't produce what you might expect. abstract class Glyph { // abstract void draw(); Glyph() { System.out.println("Glyph() before draw()"); draw(); System.out.println("Glyph() after draw()"); } } class RoundGlyph extends Glyph { int radius = 1; RoundGlyph(int r) { radius = r; System.out.println("RoundGlyph.RoundGlyph(), radius = " + radius); } void draw() { System.out.println("RoundGlyph.draw(), radius = " + radius); } } public class PolyConstructors { public static void main(String[] args) { new RoundGlyph(5); } } // /:~
代码执行的效果如下:
Glyph() before draw() RoundGlyph.draw(), radius = 0 Glyph() after draw() RoundGlyph.RoundGlyph(), radius = 5
为什么会得到上面的结果呢?
new RoundGlyph(5); 的执行顺序:RoundGlyph继承了Glyph 那么就需要先初始化Glyph以后再初始化RoundGlyph 这个初始化过程就按照前面总结的对象创建过程一样 那么为什么会
出现radius = 0的情况呢 ?
是因为在采取其他任何操作之前,为对象分配的存储空间初始化成二进制零。那么重新总结一下初始化的顺序:
(1) 在采取其他任何操作之前,为对象分配的存储空间初始化成二进制零。
(2) 就象前面叙述的那样,调用基础类构建器。此时,被覆盖的draw()方法会得到调用(的确是在RoundGlyph 构建器调用之前),此时会发现radius 的值为0,这是由于步骤(1)造成的。
(3) 按照原先声明的顺序调用成员初始化代码。
(4) 调用衍生类构建器的主体。
因此得到上面输出的效果。