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) 调用衍生类构建器的主体。

   因此得到上面输出的效果。

原文地址:https://www.cnblogs.com/liaokailin/p/3670545.html