继承的理解

来源:互联网 发布:意林小小姐阅读软件 编辑:程序博客网 时间:2024/06/14 11:04

class Depend {
int i = 10;

public Depend() {
print();
i = 20;
}

void print() {
System.out.println("Depend=> " + this.i);
}
}

class Test1 extends Depend {
int i = 30;

public Test1() {
print();
super.print();
i = 40;
}

void print() {
System.out.println("Target=> " + i);
}

}

 

public class Test{
 public static void main(String[] args) {
 new Test1();
}
}

 


我们顺着初始化的顺序来说
首先程序从main方法开始执行,new Test1(),这句话就是要new一个Test1的对象,根据对象初始化的顺序,初始化子类之前必须要初始化父类,所以此时一系列的调用开始了
1,调用Test1的父类Depend类的构造函数,在调用构造函数之前,成员变量是先于构造函数初始化的,这个时候Depend里面的i已经有值了,它的值就是10,在Depend构造函数里面,我们看到的第一句是:print方法,这个print方法我们要注意,它在Depend的子类也定义了,并且此次初始化是由子类Test1发起的,所以实际上这个print方法调用的是Test1里面定义的print,而这个时候有意思的事情就出现了,此时子类还没有出生呢,因为这个时候父类才正在构造之中,所以子类中此时的i还是0,而print正好打印出的是子类的i,所以第一次输出是0;
2,父类调用完子类的print后,把父类的i赋了值20,此时父类已经完全被构造出来了,马上就要开始构造子类了.
3,同理,在调用子类的构造函数之前,子类的i被赋了初值30,然后进入子类的构造函数,此时调用的也是print,这个就非常好理解了,这个print肯定是子类自己的print方法了,此时i已经构造好,当然,此时输出的值是30;
4,下一句super.print(),这句话显示的调用了父类的print方法,而此时父类的i已经在父类的构造函数里面改为20了,所以此次调用输出20.
5,然后再把子类的i的值设为50.

在以上过程中,如果掌握好了类的初始化顺序,是比较容易知道输出结果的.还有一点要记住,JAVA里面的方法是动态绑定的,而成员却是静态绑定的.父类里面调用的print之所以会输出0,就是因为print实际上调用的是子类的print,因为整个这场调用都是由new Test1()这句话产生的.