偶然间看到这样一道题目(来自这位作者的博客):
public class Base
{
private String baseName = "base";
public Base()
{
callName();
}
public void callName()
{
System. out. println(baseName);
}
static class Sub extends Base
{
private String baseName = "sub";
public void callName()
{
System. out. println (baseName) ;
}
}
public static void main(String[] args)
{
Base b = new Sub();
}
}
经过前面几篇的总结,对类从加载到对象的初始化应该有了比较好的认识。但对于构造函数和类的实例变量间的初始化顺序还不清楚。这里应该是:先调用父类构造器,再初始化类实例变量,最后执行构造函数里面的代码。
有了这个原则之后还是做错了,是因为没有考虑到在父类的构造函数中调用方法的多态性,想当然了,因为对普通方法的调用是invokevirtual指令,要先确定发送消息的对象,这里应该是this指针,而这个this指针应该是在main()中初始分配的子类对象的地址,所以会在子类的虚方法表中找到子类的callName的入口地址。这是一个多态。
了解了执行原理,再分析这些问题也就不难了。