Java 多態(tài)
多態(tài)是同一個行為具有多個不同表現(xiàn)形式或形態(tài)的能力
多態(tài)就是同一個接口,使用不同的實例而執(zhí)行不同操作
多態(tài)性是對象多種表現(xiàn)形式的體現(xiàn)
同一個事件發(fā)生在不同的對象上會產生不同的結果
多態(tài)的優(yōu)點
- 消除類型之間的耦合關系
- 可替換性
- 可擴充性
- 接口性
- 靈活性
- 簡化性
多態(tài)存在的三個必要條件
- 繼承
- 重寫
- 父類引用指向子類對象
比如
Parent p = new Child();
當使用多態(tài)方式調用方法時,首先檢查父類中是否有該方法误褪,如果沒有匙铡,則編譯錯誤艇肴;如果有华糖,再去調用子類的同名方法
多態(tài)的好處:可以使程序有良好的擴展,并可以對所有類的對象進行通用處理
下面的代碼演示了多態(tài)的實現(xiàn)和應用
public class Test {
public static void main(String[] args) {
show(new Cat()); // 以 Cat 對象調用 show 方法
show(new Dog()); // 以 Dog 對象調用 show 方法
Animal a = new Cat(); // 向上轉型
a.eat(); // 調用的是 Cat 的 eat
Cat c = (Cat)a; // 向下轉型
c.work(); // 調用的是 Cat 的 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 代碼子檀,輸出結果如下
吃魚
抓老鼠
吃骨頭
看家
吃魚
抓老鼠
虛方法
重寫可以讓子類能夠重寫父類的方法
當子類對象調用重寫的方法時,調用的是子類的方法乃戈,而不是父類中被重寫的方法
要想調用父類中被重寫的方法褂痰,則必須使用關鍵字 super
首先,我們先定義一個 Employee 類
public class Employee {
private String name;
private String address;
private int number;
public Employee(String name, String address, int number) {
System.out.println("Employee 構造函數(shù)");
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;
}
}
然后寫一個 Salary 類繼承 Employee 類
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;
}
}
然后寫一個測試類生成 Salary 類對象并調用一些方法
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();
}
}
編譯運行以上 Java 代碼,輸出結果如下
Employee 構造函數(shù)
Employee 構造函數(shù)
使用 Salary 的引用調用 mailCheck --
Salary 類的 mailCheck 方法
郵寄支票給:員工 A 谍憔,工資為:3600.0
使用 Employee 的引用調用 mailCheck--
Salary 類的 mailCheck 方法
郵寄支票給:員工 B 匪蝙,工資為:2400.0
在上面 3 個文件中
- 實例化了兩個 Salary 對象:一個使用 Salary 引用 s,另一個使用 Employee 引用 e
- 當調用 s.mailCheck() 時韵卤,編譯器在編譯時會在 Salary 類中找到 mailCheck()骗污,執(zhí)行過程 JVM 就調用 Salary 類的 mailCheck()
- 因為 e 是 Employee 的引用,所以調用 e 的 mailCheck() 方法時沈条,編譯器會去 Employee 類查找 mailCheck() 方法
- 在編譯的時候需忿,編譯器使用 Employee 類中的 mailCheck() 方法驗證該語句, 但是在運行的時候,Java虛擬機(JVM)調用的是 Salary 類中的 mailCheck() 方法
這個過程被稱為虛擬方法調用屋厘,該方法被稱為虛擬方法
Java 中所有的方法都能以這種方式表現(xiàn)涕烧,因此,重寫的方法能在運行時調用汗洒,不管編譯的時候源代碼中引用變量是什么數(shù)據類型