Java 多態
多態是同一個行為具有多個不同表現形式或形態的能力。
多態就是同一個介面,使用不同的實例而執行不同操作,如圖所示:

多態性是對象多種表現形式的體現。
現實中,比如我們按下 F1 鍵這個動作:
-
如果當前在 Flash 介面下彈出的就是 AS 3 的幫助文檔;
- 如果當前在 Word 下彈出的就是 Word 幫助;
- 在 Windows 下彈出的就是 Windows 幫助和支持。
同一個事件發生在不同的對象上會產生不同的結果。
多態的優點
-
1. 消除類型之間的耦合關係
-
2. 可替換性
-
3. 可擴充性
-
4. 介面性
-
5. 靈活性
-
6. 簡化性
多態存在的三個必要條件
比如:
Parent p = new Child();
當使用多態方式調用方法時,首先檢查父類中是否有該方法,如果沒有,則編譯錯誤;如果有,再去調用子類的同名方法。
多態的好處:可以使程式有良好的擴展,並可以對所有類的對象進行通用處理。
以下是一個多態實例的演示,詳細說明請看注釋:
Test.java 檔代碼:
public class Test {
public static void main(String[] args) {
show(new Cat());
show(new Dog());
Animal a = new Cat();
a.eat();
Cat c = (Cat)a;
c.work();
}
public static void show(Animal a) {
a.eat();
if (a instanceof Cat) {
Cat c = (Cat)a;
c.work();
} else if (a instanceof Dog) {
Dog c = (Dog)a;
c.work();
}
}
}
abstract class Animal {
abstract void eat();
}
class Cat extends Animal {
public void eat() {
System.out.println("吃魚");
}
public void work() {
System.out.println("抓老鼠");
}
}
class Dog extends Animal {
public void eat() {
System.out.println("吃骨頭");
}
public void work() {
System.out.println("看家");
}
}
執行以上程式,輸出結果為:
吃魚
抓老鼠
吃骨頭
看家
吃魚
抓老鼠
虛函數
虛函數的存在是為了多態。
Java 中其實沒有虛函數的概念,它的普通函數就相當於 C++ 的虛函數,動態綁定是Java的默認行為。如果 Java 中不希望某個函數具有虛函數特性,可以加上 final 關鍵字變成非虛函數。
重寫
我們將介紹在 Java 中,當設計類時,被重寫的方法的行為怎樣影響多態性。
我們已經討論了方法的重寫,也就是子類能夠重寫父類的方法。
當子類對象調用重寫的方法時,調用的是子類的方法,而不是父類中被重寫的方法。
要想調用父類中被重寫的方法,則必須使用關鍵字 super。
Employee.java 檔代碼:
public class Employee {
private String name;
private String address;
private int number;
public Employee(String name, String address, int number) {
System.out.println("Employee 構造函數");
this.name = name;
this.address = address;
this.number = number;
}
public void mailCheck() {
System.out.println("郵寄支票給: " + this.name
+ " " + this.address);
}
public String toString() {
return name + " " + address + " " + number;
}
public String getName() {
return name;
}
public String getAddress() {
return address;
}
public void setAddress(String newAddress) {
address = newAddress;
}
public int getNumber() {
return number;
}
}
假設下麵的類繼承Employee類:
Salary.java 檔代碼:
public class Salary extends Employee
{
private double salary;
public Salary(String name, String address, int number, double salary) {
super(name, address, number);
setSalary(salary);
}
public void mailCheck() {
System.out.println("Salary 類的 mailCheck 方法 ");
System.out.println("郵寄支票給:" + getName()
+ " ,工資為:" + salary);
}
public double getSalary() {
return salary;
}
public void setSalary(double newSalary) {
if(newSalary >= 0.0) {
salary = newSalary;
}
}
public double computePay() {
System.out.println("計算工資,付給:" + getName());
return salary/52;
}
}
現在我們仔細閱讀下麵的代碼,嘗試給出它的輸出結果:
VirtualDemo.java 檔代碼:
public class VirtualDemo {
public static void main(String [] args) {
Salary s = new Salary("員工 A", "北京", 3, 3600.00);
Employee e = new Salary("員工 B", "上海", 2, 2400.00);
System.out.println("使用 Salary 的引用調用 mailCheck -- ");
s.mailCheck();
System.out.println("\n使用 Employee 的引用調用 mailCheck--");
e.mailCheck();
}
}
以上實例編譯運行結果如下:
Employee 構造函數
Employee 構造函數
使用 Salary 的引用調用 mailCheck --
Salary 類的 mailCheck 方法
郵寄支票給:員工 A ,工資為:3600.0
使用 Employee 的引用調用 mailCheck--
Salary 類的 mailCheck 方法
郵寄支票給:員工 B ,工資為:2400.0
例子解析
-
實例中,實例化了兩個 Salary 對象:一個使用 Salary 引用 s,另一個使用 Employee 引用 e。
-
當調用 s.mailCheck() 時,編譯器在編譯時會在 Salary 類中找到 mailCheck(),執行過程 JVM 就調用 Salary 類的 mailCheck()。
-
因為 e 是 Employee 的引用,所以調用 e 的 mailCheck() 方法時,編譯器會去 Employee 類查找 mailCheck() 方法 。
-
在編譯的時候,編譯器使用 Employee 類中的 mailCheck() 方法驗證該語句,
但是在運行的時候,Java虛擬機(JVM)調用的是 Salary 類中的 mailCheck() 方法。
以上整個過程被稱為虛擬方法調用,該方法被稱為虛擬方法。
Java中所有的方法都能以這種方式表現,因此,重寫的方法能在運行時調用,不管編譯的時候源代碼中引用變數是什麼數據類型。
多態的實現方式
方式一:重寫:
這個內容已經在上一章節詳細講過,就不再闡述,詳細可訪問:
Java 重寫(Override)與重載(Overload)。
方式二:介面
方式三:抽象類和抽象方法
詳情請看 Java抽象類 章節。