我有以下2个班级
public class classA { classA() { System.out.println("A"); } } class classB extends classA { classB() { System.out.println("B"); } }
然后跑
1classA c = new classB();
要么
2classB c = new classB();
总是给
A B
为什么会这样?乍一看,在任何一种情况下,我都会假设只classB
调用构造函数,因此唯一的输出就是
B
但这显然是错误的.
这就是Java的工作原理.在调用Object
子类的构造函数之前,调用父类的构造函数,一直到类层次结构.
引用文档:
使用
super()
,调用超类无参数构造函数.使用super(parameter list)
,调用具有匹配参数列表的超类构造函数.注意:如果构造函数未显式调用超类构造函数,则Java编译器会自动插入对超类的无参数构造函数的调用.如果超类没有无参数构造函数,则会出现编译时错误.
Object
确实有这样的构造函数,所以如果Object
是唯一的超类,则没有问题.如果子类构造函数显式或隐式地调用其超类的构造函数,您可能会认为将调用一整个构造函数链,一直回到构造函数
Object
.事实上,情况就是这样.它被称为构造函数链接,当需要很长的类下降时,您需要注意它.
在构造过程中总是调用超类构造函数,并且保证在调用子类构造函数之前完成超类构造.对于大多数(如果不是全部)面向对象语言来说就是这种情况.如果您不想调用默认构造函数,可以使用参数显式调用超类构造函数; 否则这样的调用由编译器自动完成.