今天碰到了个构造器的问题,在此谨记。如下所示,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 constructor
Member constructor
Child constructor
结论
- 构造的时候首先会调用父类的构造器,构造父类的各个成员
- 然后初始化Child类里的像member一样的成员,这些成员都是直接new的形式初始化的
执行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根本就没有构造完全