java对象构造过程

来源:互联网 发布:有约束的最短路径算法 编辑:程序博客网 时间:2024/05/20 06:25

今天碰到了个构造器的问题,在此谨记。如下所示,Child类继承自Parent。那么一个Child对象内部有哪些东西呢?首先他既然是继承自Parent,那么他必然有Parent内部的各个成员。然后他有自己的成员,比如y和member。

对象构造

那么这些成员是怎么初始化的?

public class Child extends Parent {    private Member member = new Member();    private int y;    public Child(int x) {        super(x);        //此时父类已经构造好了,下一步构造成员变量比如member        LogUtil.fish("Child constructor");        y = 10;    }    @Override    protected void lala() {        LogUtil.fish("Child lala");    }}

parent的代码如下

public class Parent {    int x;    public Parent(int x) {        LogUtil.fish("Parent constructor");        this.x = x;    }}

执行日志为

Parent constructorMember constructorChild constructor

结论

  1. 构造的时候首先会调用父类的构造器,构造父类的各个成员
  2. 然后初始化Child类里的像member一样的成员,这些成员都是直接new的形式初始化的
  3. 执行Child构造函数的剩余部分,比如下边的y。

    所以看起来member的构造函数就插在super(x)的后面

构造过程执行多态

我们再看看父类,父类在构造的过程中,调用了lala,而lala是多态的。所以在

public class Child extends Parent {    private Member member = new Member();    private int y;    public Child(int x) {        super(x);        //此时父类已经构造好了,下一步构造成员变量比如x        LogUtil.fish("Child constructor");        y = 10;    }    @Override    protected void lala() {        LogUtil.fish("Child lala");        member.print();    }}

parent的代码如下

public class Parent {    int x;    public Parent(int x) {        LogUtil.fish("Parent constructor");        this.x = x;        lala();    }    protected void lala() {        LogUtil.fish("Parent lala");    }}

这里会发生crash,看的出来吗
在parent构造的过程中,多态调用了子类的lala,子类的lala内有member.print();,但是此时member没有初始化(只有在父类构造完毕之后member才初始化),所以此时member为空,直接NPE挂了。我遇到的bug比这个复杂很多,调用没这么明显,所以查了好久才查清楚。
我当时就想,member怎么可能为null,事实证明member的确有可能为空的,在执行lala的时候child根本就没有构造完全

原创粉丝点击