java多态的内存图解

版权声明:本文为博主原创文章,未经博主容许不得转载。android

图解Java多态内存分配以及多态中成员方法的特色
spa

Person worker = new Worker(); 子类实例对象地址赋值给父类类型引用变量。多态的体现。.net


多态中成员方法的特色分析:对象


【子类有,父类没有】编译失败!!!
worker.startWork(); 为何编译不经过呢?提示:找不到符号。
由于引用变量worker是Person类型,在Person类的方法表中查找方法startWork(),找获得吗?找不到,更别提常量池解析了。编译失败。


【子类有,父类有,重写,非静态】调用子类!!!
worker.say(); 子类重写父类方法,被重写的方法在子类跟父类的方法表中索引相同。
调用的时候,在父类类型的方法表中查找say方法的索引,而后把索引存到PolDemo类的常量表中(常量池解析,就是用Person类方法表中的索引项来代替常量池中的符号引用)。
由于索引相同,直接拿常量表中say方法的索引去子类方法表中调用say()方法。 因此,此时调用的是被子类重写的方法。见图中的内存分配。


【子类有,父类有,静态】调用当前引用类型变量类型中的方法。
由于静态是属于类的,由实例共享,因此只看当前引用变量所属的类中的静态方法。


多态中(父类引用指向子类对象)成员方法(非静态)有如下特色:
编译期根据引用类型变量所属的类型方法表中是否有调用的方法,没有编译失败。
运行期根据在堆中建立对象的实际类型找到对应的方法表,从中肯定具体的方法在内存中的位置。


堆中实例对象:子类包含父类,子类对父类说:你的就是个人,个人仍是个人。


多态中成员变量的特色分析:
不管编译期仍是运行期,都只参考引用类型变量所属的类中是否有对象的成员变量。


小的总结:
再来看这条语句:Person worker = new Worker(): 多态,父类引用指向子类对象。
跟这句代码的做用是同样的,Worker w = new Worker(); Person worker = w; 就是子类的引用传给父类类型的引用。向上转型,都指向一个实例对象(子类)。
1:为何能够将子类实例对象的引用传给父类类型引用呢? 
答:多态的体现,鸽子是鸟类,鹦鹉是鸟类,喜鹊是鸟类,它们都是鸟类,这就是鸟类的多种表现形态(多态),
它们有鸟类的基本特征与行为,而且还有本身特有的行为,那就会把鸟类的基本特征与行为继承过来给本身。extends关键字声明了他们的关系。
程序中这句话 鸟类 xx鸟 = new 鸽子();就建立了一个鸟类的引用,它是鸽子。
类的概念是抽象的,因此它就仅仅是个引用,只有引用到实例对象,它才真正实现了本身。


2:父与子的关系会怎样呢?
因为Worker类继承自Person类,因此会先加载Person类并初始化成员变量在Worker类实例对象中。这就是子类继承过来的成员变量。
若是子类中定义了与父类相同的成员变量,变量不会存在重写,子类实例对象中会有两份相同名称的实例变量,调用时,父类的会被隐藏。
若是想调用父类被隐藏的成员变量,那就要使用super关键字。
一样,子类会继承父类的成员方法做为本身的成员方法,若是子类定义了与父类相同的成员方法(重写),多态中会调用子类本身的成员方法。


3:多态,又有什么好处呢?跟这样写(Worker worker = new Worker();)有什么区别,不同继承父类的属性方法吗?
首先明确: Worker worker = new Worker();这样写仅仅是继承关系。不存在多态特性。
Person worker = new Worker(); 这样写就展现了多态的关系,鸽子是鸟类的一种表现形态。
有啥好处?一句话归纳:提升了程序的扩展性,表如今什么地方呢?好比:
如今有个工厂,能解析鸟类的语言,鸽子是鸟类吧?Ok,那建一个工厂,让鸽子说。
public static void doWork(Dove dove) {
String sound = dove.say(); // ... 拿到鸽子说的话
}
鹦鹉是鸟类吧?Ok,那建一个工厂,让鹦鹉说。
public static void doWork(Parrot parrot) {
String sound = parrot.say(); // ... 拿到鹦鹉说的话
}
若是还有一千个鸟类要说话,那我岂不得建一千座工厂?。。。累不累?那既然都是鸟类,我建一个鸟类的工厂不就好了嘛?
public static void doWork(Bird bird) {
String sound = bird.say(); // ... 拿到xx鸟说的话。。。
}
这时候,不管你什么鸟进来,我都让你说话,并且说得都是你本身的话,由于你继承我并重写了你的功能。
-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
Worker w = (Worker)worker; // 向下转型。目的,为了调用子类非继承父类、本身特有的方法。
由于多态有个弊端,只能使用父类的引用访问父类的成员。因此向下转型是为了访问子类本身的成员。
首先,worker引用指向的实例对象原本就是子类类型的。因此赋值给子类类型引用变量很是能够。
而后如今用子类类型的引用就能够访问本身的成员方法了啦啦啦。
blog

相关文章
相关标签/搜索