java构造器内部的多态方法

我们知道,动态绑定的调用是在运行时才决定的,对象无法知道到底调用的是哪个类的方法。
当我们在构造器中调用动态绑定的方法,就会用到该方法被覆盖之后的定义。但是这种调用的效果难以预计,因为被覆盖的方法在对象被完全构造之前就会被调用。我们先来看看下面这段代码:
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();

}
}

从代码中可以看出,子类Derived覆盖了父类Base的方法。当我们实例化一个Derived对象的时候,会自动调用父类的构造器,而父类构造器中却调用了动态绑定方法
tellName(); 

这样一来,在父类构造器中就会调用子类的tellChild()方法。此时父类Base已完成初始化。。由于此时子类Derived还没有完成初始化,其实例变量name为默认为null。然后父类构造器退出,回到子类构造器调用处,然后完成子类初始化,接下来子类构造器调用完成。
代码的运行结果如下;
tellChild null
printChild base
tellChild Drived
printChild Drived

实际的初始化过程应该是这样的:
1. 在其他任何事物发生之前,将分配给对象的存储空间初始化为二进制的零。
2. 如上文一般,调用父类构造器,在父类构造器中会调用被子类覆盖的方法(如果有在构造器中调用了动态绑定的方法)。
3. 按照声明的顺序调用成员的初始化方法。
4. 调用子类的构造器。

从上例可以看出,在构造器中能安全调用的方法是不能被继承的,即该方法被final修饰,或者方法是private的(private方法自动属于是final的,嘿嘿)。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值