我们先建立一个父类,名为Father.java,类中代码如下:
public class Father{}
再建立一个子类,名为Son.java,类中代码如下:
public class Son extends Father {}
我们知道,如果在类中没有声明构造函数,javac编译时,会自动的为我们加上一个公开的无参构造函数。
Windows7 cmd下JDK1.8.0_151环境,我们使用【javac Son.java】命令编译Son.java代码,javac会“聪明的”将Father也编译了。
于是我们可以得到Son.class和Father.class两个字节码文件。
我们先看看Father.class反编译结果,使用【javap -c Father.class】命令查看:
果不其然,javac自动的为我们添加了一个无参的构造函数。可是,细心的读者一定会奇怪的发现,这个java.lang.Object是干吗的?还用到了一个字节码指令invokespecial。它们都是干吗的? 别着急,我们先看看Son.class的反编译结果。
使用命令【Javap -c Son.class】看看Son.class反编译的结果:
同Father.class反编译结果类似,javac自动为Son.class增加了一个无参构造函数。
重点在这个构造函数里,Method Father."<init>":()V语句。是不是与Father.class中构造函数很类似,只是成了Method java/lang/Object."<init>":()V。
根据上面这些信息,我们开始思考。
同是invokespecial指令,既然Son是继承自Father,那么Father是不是以此推理,继承自java.lang.Object呢?可我们在Father类的定义中并没有声明它继承Object类啊?原因在于,java中默认所有“没有声明继承其他类的”的类都是默认继承java.lang.Object类。至于为什么要继承Object,我会在其他的文章里讲述。这里读者只需要记住这个特点就行了。
我们改造一下Father和Son的源文件,显示声明默认构造函数,看看会发生什么。
public class Father {
public Father() {}
}
public class Son extends Father {
public Son() {}
}
还是使用javac编译,得到Son.class和Father.class,这次我将两个类反编译结果同时展示出来:
是不是和上面的结果一模一样。
我把Son.java文件,再改造一下,改成这样:
public class Son extends Father() {
public Son() {
super();
}
}
再次编译Son.java,得到新的Son.class和Father.class文件,展示如下:
还是一样。我们把Son.java代码改造一下,这样写:
public class Son extends Father {
public Son() {
new Father();
}
}
然后编译Son.java文件,再反编译Son.class得到如下结果:
同上面的图片对比,new是分水岭,我们知道了Method Father."<init>":()V调用的是Father的构造函数。并且在这里调用了两次。由此我们知道了,子类执行构造函数时,会默认执行父类构造函数,并且首先执行父类默认构造函数。
【总结】:子类执行构造函数,会默认执行父类构造函数。
【思考】:如果父类中有多个构造函数呢?如果子类中有多个构造函数呢,怎么调用?
【备注】:限于篇幅,还是没有讲super和invokespecial指令是干嘛的,我会在其他文章中专门来讲解。