标签:
动态分派与复写密不可分,因为java中存在向上转型,这样就涉及到方法的调用问题。先看一下示例代码
package com.dy.xidian; class Test1 { public void say() { System.out.println("Test1"); } } public class Test extends Test1 { public static void main(String[] args) { Test1 t = new Test(); t.say(); } public void say() { System.out.println("Test"); } }
代码中创建了一个Test()对象,然后将其向上转型为Test1类型,然后调用say()方法。那么问题来了,调用的say()是父类的方法还是子类的方法。先看一下字节码文件
在字节码文件中,我们可以观察到在main()方法中创建了Test文件,然后调用构造器(<init>)方法,紧接着调用Test1.say()方法。那么最后应该输出的是Test1,但是运行上面的代码会发现输出的是Test。出现这种原因和invokevirtual指令有关。
invokevirtual指令执行过程
所以上面代码中t是静态类型,编译时就可以确认,为Test1。而其对应的对象是实际类型,为Test。所以按照上面的步骤最后的结果输出的应该是Test。
标签:
原文地址:http://www.cnblogs.com/xidongyu/p/5915522.html