时间久了不用就会忘记,特此记录一下
文中部分引用内容来自:java提高篇(四)-----理解java的三大特性之多态
一、多态的定义
多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。因为在程序运行时才确定具体的类,这样,不用修改源程序代码,就可以让引用变量绑定到各种不同的类实现上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态,这就是多态性。
二、举例
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
public class IteratorDemo {
public static void main(String[] args) {
List list = new ArrayList();
list.add(1);
list.add(4);
list.add(8);
Iterator<Integer> iter = list.listIterator();
int oldValue = iter.next();
System.out.println(oldValue);
((ListIterator) iter).set(10);
System.out.println(iter.equals(10));
System.out.println(iter.next());
System.out.println(iter.next());
((ListIterator) iter).set(11);
System.out.println(list);
}
}
通过例子来讲一下多态.
对于set方法.Iterator类中没有此方法,但子类拥有set方法.
Iterator<Integer> iter = list.listIterator();
使用了多态.
iter.set(10);
会报错The method set(int) is undefined for the type Iterator<Integer>
:Iterator类中没有set(int)方法的定义.
查阅文档后发现Iterator父类中并没有set(int)方法.此时就产生疑惑了,此处不是应用多态了么?怎么还会有错误呢?直接使用子类的set方法不就可以了么?结果显示是不可以的,原因就是多态的规定里面的第三条
多态的规定:
- 当父类定义的方法而子类没有重写该方法,则直接使用父类的方法即可.
- 当父类定义的方法子类重写了该方法,则调用该方法时将会使用子类重写的方法.
- 当父类未定义的方法,但子类却单独拥有(就算子类重载父类的方法(重载后的方法与原方法属于两个不同的方法)也属于这种情况),这样的情况(也就是该例子遇到的情况),指向子类的父类是无法使用子类独有的方法的.且会在编译时报错父类并未定义该方法.
三、重写时满足的条件
引自 – <Java中可变长参数的使用及注意事项>
(1)重写方法不能缩小访问权限;(如果父类是public,那么子类必须是public)
(2)参数列表必须与被重写方法相同(包括显示形式);
(3)返回类型必须与被重写方法的相同或是其子类;
(4)重写方法不能抛出新的异常,或者超过了父类范围的异常,但是可以抛出更少、更有限的异常,或者不抛出异常。
四、多态的总结
1. 指向子类的父类引用由于向上转型了,它只能访问父类中拥有的方法和属性
2. 而对于子类中存在而父类中不存在的方法,该引用是不能使用的,尽管是重载该方法(重载后的方法属于两个不同的方法)。
3. 若子类重写了父类中的某些方法,在调用该些方法的时候,必定是使用子类中定义的这些方法(动态连接、动态调用)。
public class Demo {
public static void main(String[] args) {
// 指向子类的父类由于向上转型了, 它只能访问父类中拥有的方法和属性
Animal a = new Cat();
// 子类重写的父类方法, 将会调用子类重写的方法
a.eat();
a.sleep();
// 想要调用子类特有的方法的话, 需要向下转型.
// 在向下转型时, 需要进行类型的判断
// 判断变量a中如果保存的是Dog类型的对象的地址值
if (a instanceof Dog) {
Dog dog = (Dog) a;
a.lookHome();
} else if (a instanceof Cat) {
Cat cat = (Cat) a;
a.catchMouse();
}
}
}