我们知道,动态绑定的调用是在运行时才决定的,对象无法知道到底调用的是哪个类的方法。
printChild base
tellChild Drived
printChild Drived
当我们在构造器中调用动态绑定的方法,就会用到该方法被覆盖之后的定义。但是这种调用的效果难以预计,因为被覆盖的方法在对象被完全构造之前就会被调用。
class Base{
private String name = "base";
Base(){
tellName();
printName(name);
}
public void tellName(){
System.out.println("tellFather "+ name);
}
public void printName(String name){
System.out.println("printFather "+ name);
}
}
class Drived extends Base{
private String name = "Drived";
Drived(){
tellName();
printName(name);
}
public void tellName(){
System.out.println("tellChild "+ name);
}
public void printName(String name){
System.out.println("printChild "+ name);
}
}
public class TestObject {
public static void main(String args[]){
new Drived();
}
}
代码运行结果
tellChild nullprintChild base
tellChild Drived
printChild Drived
初始化过程
1. 在其他任何事物发生之前,将分配给对象的存储空间初始化为二进制的零。
2. 如上文一般,调用父类构造器,在父类构造器中会调用被子类覆盖的方法(如果有在构造器中调用了动态绑定的方法)。
3. 按照声明的顺序调用成员的初始化方法。
4. 调用子类的构造器。
讨论
从上例可以看出,在构造器中能安全调用的方法是不能被继承的,即该方法被final修饰,或者方法是private的(private方法自动属于是final的)。