先执行父类静态成员,再子类静态成员,然后父类实例成员和构造器,最后子类实例成员和构造器;若父类构造器调用被重写的方法,会触发多态,但子类字段未初始化,导致输出0;应避免在构造器中调用可重写方法。
在Java中,类的初始化顺序对多态行为有着直接影响,尤其是在涉及继承和方法重写时。理解这一过程有助于避免运行时的意外结果。
Java对象的初始化遵循固定顺序:
这个顺序是固定的,无论是否存在多态调用。关键问题是:如果在构造器中调用了被子类重写的方法,会发生什么?
当父类构造器调用一个被子类重写的方法时,会触发多态行为,即实际执行的是子类中的版本。但此时子类的初始化可能尚未完成,导致访问未正确初始化的字段。
示例说明:
class Parent {
protected int value = 10;
public Parent() {
print(); // 多态调用
}
public void print() {
System.out.println("Parent: " + value);
}
}
class Child extends Parent {
private int value = 20;
public Child() {
super();
}
@Override
public void print() {
System.out.println("Child: " + value);
}
}
执行 new Child() 时输出为:
Child: 0
原因在于:虽然调用了子类的 print 方法,但此时子类的 value 字段还未初始化(要等父类构造器执行完才轮到子类实例变量赋值),所以取到的是默认值 0。
为防止此类问题,应遵循以下原则:
基本上就这些。只要意识到构造过程中多态仍然生效,且子类状态可能未就绪,就能写出更安全的继承代码。