public class Fruit {
protect String name = "fruit";
public void print(){
System.err.print(name);
}
}
public class Apple extends Fruit {
protect String name = "apple";
}
public class Client {
public static void main(String[] args) {
Apple apple = new Apple();
apple.print();
}
}
你认为 apple.print 会输出哪一种(为什么):
1: fruit
2:apple
内存模型大概示意图如下:
------------------------------------------------------ 分割线 --------------------------------------------------------------------------------
得出结论如下:
1、若子类覆盖了某方法,则父类引用调用子类重新定义的新方法
2、若子类未覆盖某方法,则父类引用调用父类本身的旧方法
3、若子类覆盖了某属性,但父类引用仍调用父类本身的旧属性
4、若子类未覆盖某属性,则父类引用调用父类本身的旧属性
继承是如何实现,内部实现原理是怎么样的?
1、每个对象在调用方法的时候,实际上方法存在一个隐式参数,例如:
Apple mg=
new
Apple();
apple.print(); //print实际上存在一个隐式参数,当前对象的引用,执行是会将对象引用传
进去,变成apple.print(mg);
2、每个类,虚拟机都会为每个类预先创建一个方法表(method table),用来记录所有方法所对应的内存地址。在调用方法的时候,直接去找方法对应的内存地址进行执行,如果是单纯的继承,相当于默认给子类创建一个跟父类类似的method table。
因此,对于不同的子类对象调用子类方法时,会从子类方法表中,查询该方法的地址,如果子类覆盖父类方法,那么对应的方法的地址也被修改了,进而实现多态。
参考文章:
1、https://my.oschina.net/u/1760791/blog/769842
2、http://blog.51cto.com/hssmy/1928421