java不想C++中的构造器,java的构造器可以准许你使用多态进行动态绑定。
动态绑定是在运行时决定的,因为对象无法知道他是属于那一个类,这就使构造器出现了两难的问题:如果构造器中正在调用构造器所在的类需要动态绑定的方法,那会发生什么情况呢?答案是:构造器中调用的方法会用到那个方法被覆盖后的定义(可能招致灾难),而父类构造器没有初始化完成,就不可能初始化完成子类,如果在方法中访问一些子类变量,可能发生你不想要的结果从而招致灾难发生。
class Father{
void f(){
System.out.println("父类f()");
}
public Father() {
System.out.println("before");
f();
System.out.println("after");
}
}
class Son extends Father{
int i=1;
void f(){
<span style="white-space:pre"> </span>System.out.println("子类f(); "+"i的值是"+i);
}
}
public class Test{
public static void main(String[] args){
new Son();
}
}
before
子类f(); i的值是0
after
从输出结果来看 ,可以看出在java构造器中如果使用动态绑定的方法,则调用的是覆盖后的方法。
之所以i的输出值是0,是由于初始化的顺序如下:
1.在其他任何事物发生之前,将分配给对象的存储空间初始化为二进制的0,。
2.调用基类构造器,此时访问的i的值为0.
3.然后按照生命顺序初始化变量。
4.调用导出类的构造器的主体。
所以,确切的说,在构造器中只有调用基类的final的方法时安全的,因为它不能被覆盖,就不会出现上述问题。