6 个回答

首先,你的代码和下面代码是等价的:

class X {

    Y b;

    public X() {
        b = new Y();
        System.out.println("X");
    }

}

class Y {

    public Y() {
        System.out.println("Y");
    }

}

public class Z extends X {

    Y y;

    public Z() {
        y = new Y();
        System.out.println("Z");
    }

    public static void main(String[] args) throws Exception {
        new Z();
    }
}
  1. new Z() 之后,调用 Z 的无参构造函数;

  2. 因为 Z 继承了 X ,所以去调用 X 的无参构造函数(执行 super() );

  3. X 的无参构造函数第一行代码为 Y b = new Y(),所以去调用 Y 的无参构造函数,此时输出 Y

  4. Y b = new Y() 执行完毕之后,回到 X 的无参构造函数,输出 X

  5. 然后回到 Z 的无参构造函数,执行Y y = new Y() 输出 Y,最后输出 Z

所以图中提到的 先执行 X 的构造函数 的说法是错误的,应该最先执行的是 Y 的构造函数。

函数的声明 void可以省略?

运行结果是YXYZ,有什么不对吗?

对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序依次是(静态变量、静态初始化块)>(变量、初始化块、构造器。)

继承与初始化分为三步 (p273):

1 找到所有的基类从根基类开始加载 
2.根基类的static初始化然后是下一个导出类 以此类推 
3.所有的类都已经加载完 对象可以被创建了,首先对象中所有基本类型都会被设为默认值,对象引用都被设为null,赋初值,构造器从根往下执行赋值 

顺序:

父类的静态成员变量和静态代码块->子类的静态成员变量和静态代码块->父类成员变量和方法块->父类的构造函数->子类成员变量和方法块->子类的构造函数

            

宣传栏