一、昨天晚上跟室友瞎扯淡时谈论到java多态,然后自己默默的去网上搜索了下,看到下面这篇文章,自己看不懂,仔细看下作者的分析才明白,又get了一个新技能,于是就记录下来:
什么是多态?
多态分两种:
(1) 编译时多态(设计时多态):方法重载。
(2) 运行时多态:JAVA运行时系统根据调用该方法的实例的类型来决定选择调用哪个方法则被称为运行时多态。(我们平时说得多的事运行时多态,所以多态主要也是指运行时多态)
运行时多态存在的三个必要条件:
一、要有继承(包括接口的实现);
二、要有重写;
三、父类引用指向子类对象。
--------------------------------------------------------------------------------
详细解释:
运行时多态的解释:a.运行时多态是指程序中定义的引用变量所指向的具体类型和b.通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定.
1.程序序中定义的引用变量所指向的具体类型不确定(即一个引用变量倒底会指向哪个类的实例对象) 。
例子 :
driver 类中 drive 方法 (Vehicle类 vehicle){}
•oneDriver.drive( new car() )
•oneDriver.drive( new bus() )
其中vehicle 变量无法确定具体使用哪个子类实例。
1.通过该引用变量发出的方法调用在编程时并不确定(该引用变量发出的方法调用到底是哪个类中实现的方法) 。
例子 : 厨师,园丁,理发师的Cut 方法调用.persion.cut().
--------------------------------------------------------------------------------
多态的好处:
1.可替换性(substitutability)。多态对已存在代码具有可替换性。例如,多态对圆Circle类工作,对其他任何圆形几何体,如圆环,也同样工作。
2.可扩充性(extensibility)。多态对代码具有可扩充性。增加新的子类不影响已存在类的多态性、继承性,以及其他特性的运行和操作。实际上新加子类更容易获得多态功能。例如,在实现了圆锥、半圆锥以及半球体的多态基础上,很容易增添球体类的多态性。
3.接口性(interface-ability)。多态是超类通过方法签名,向子类提供了一个共同接口,由子类来完善或者覆盖它而实现的。如图8.3 所示。图中超类Shape规定了两个实现多态的接口方法,computeArea()以及computeVolume()。子类,如Circle和Sphere为了实现多态,完善或者覆盖这两个接口方法。
4.灵活性(flexibility)。它在应用中体现了灵活多样的操作,提高了使用效率。
5.简化性(simplicity)。多态简化对应用软件的代码编写和修改过程,尤其在处理大量对象的运算和操作时,这个特点尤为突出和重要。
实际运用:
结合配置文件的使用,联系Spring框架,利用反射,动态的调用类,同时不用修改源代码,直接添加新类和修改配置文件,不需要重启服务器便可以扩展程序。
--------------------------------------------------------------------------------
小结:
使用父类类型的引用指向子类的对象,该引用调用的师父类中定义的方法和变量,变量不能被重写(覆盖);如果子类中重写了父类中的一个方法,那么在调用这个方法的时候,将会调用子类中的这个方法;
注意特殊情况,如果该父类引用所调用的方法参数列表未定义,就调用该父类的父类中查找,如果还没找到就强制向上类型转换参数列表中的参数类型,具体优先级高到低依次如下:
this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。
--------------------------------------------------------------------------------
经典笔试题(混合重载和重写):
(一)相关类
class A {
public String show(D obj)...{
return ("A and D");
}
public String show(A obj)...{
return ("A and A");
}
}
class B extends A{
public String show(B obj)...{
return ("B and B");
}
public String show(A obj)...{
return ("B and A");
}
}
class C extends B...{}
class D extends B...{}
(二)问题:以下输出结果是什么?只是方法是在A中实现而已);现在子类B覆盖了show(A obj),因此它最终锁定到B的show(A obj)。这就是那句话的意义所在。
主要抓住几个关键词:优先级由高到低依次为:this.show(O)、super.show(O)、
this.show((super)O)、super.show((super)O);
加上理解这句话:当超类对象引用变量(A a2)引用子类对象(new B())时,被引用对象的类型(类型为:B)而不是引用变量的类型(类型为:A)决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。(A a=new B()调用引用子类对象的方法必须在引用变量超类A中定义过)
如果没有定义而是子类中的一个新方法,则不会直接调用引用子类对象的方法,而是按照上面的优先级由高到低依次为:this.show(O)、super.show(O)、
this.show((super)O)、super.show((super)O)来进行分析、调用。
转载地址,请点击打开