下面将讲讲Java里子类继承父类,然后创建子类对象时,子类的构造函数初始化过程。
如下面代码:
public class TestDemo2 {
public static void main(String[] asgr) {
B b = new B();
}
}
class A {
A(){
way();
}
void way(){
System.out.println("父类A中的way函数");
}
}
class B extends A {
int c = 15;
B(){
way();
}
void way(){
System.out.println("子类B中的way函数:"+"c="+c);
}
}
代码执行结果:
代码解析:
1)在创建B的对象时,也就是new B()时,会找到类名为B的构造器进行初始化。
2)找到构造器后,第一步,执行super(),为什么会是执行super()呢,其实在每个构造器里的第一句默认都是super(),你不写,编译器会默认加上,如:
B(){
super();//不写编译器会默认加上
way();
}
3)执行super()找到父类A的构造器进行初始化,执行父类A的构造器中的代码,也就是way()方法,这里就要说明一下了,当父类A中的way()方法与子类B中的way()方法同名时,父类A的构造器中的way()方法并不是父类A中的way()方法,而是子类B中的way()方法,因为创建的是子类B的对象,所以父类A的构造器中的方法在与子类B的方法同名时,先找的是子类B的方法,简单点来说就是如下:
A(){
this.way();//this编译器也会默认加上
}
在类中所有的对象成员其实都是有默认this的,而父类A的构造器中的this其实指向的是子类B的对象,所以说父类A构造器中的way()方法其实是调用了子类B的way()方法。
4)在父类A的构造器初始化好后,得到的结果就是下面这个:
5)然后开始初始化子类B的成员变量了,也就是代码中的
int c = 15;
6)最后才执行子类B的way()方法,所以得到的结果是:
7)在执行完上面的所有初始化后,对象才把首地址值给类变量b。