# Java 多態
多態是同一個行為具有多個不同表現形式或形態的能力。
多態性是對象多種表現形式的體現。
比如我們說"寵物"這個對象,它就有很多不同的表達或實現,比如有小貓、小狗、蜥蜴等等。那么我到寵物店說"請給我一只寵物",服務員給我小貓、小狗或者蜥蜴都可以,我們就說"寵物"這個對象就具備多態性。
接下來讓我們通過實例來了解Java的多態。
### 例子
```
public interface Vegetarian{}
public class Animal{}
public class Deer extends Animal implements Vegetarian{}
```
因為Deer類具有多重繼承,所以它具有多態性。以上實例解析如下:
* 一個 Deer IS-A(是一個) Animal
* 一個 Deer IS-A(是一個) Vegetarian
* 一個 Deer IS-A(是一個) Deer
* 一個 Deer IS-A(是一個)Object
在Java中,所有的對象都具有多態性,因為任何對象都能通過IS-A測試的類型和Object類。
訪問一個對象的唯一方法就是通過引用型變量。
引用型變量只能有一種類型,一旦被聲明,引用型變量的類型就不能被改變了。
引用型變量不僅能夠被重置為其他對象,前提是這些對象沒有被聲明為final。還可以引用和它類型相同的或者相兼容的對象。它可以聲明為類類型或者接口類型。
當我們將引用型變量應用于Deer對象的引用時,下面的聲明是合法的:
```
Deer d = new Deer();
Animal a = d;
Vegetarian v = d;
Object o = d;
```
所有的引用型變量d,a,v,o都指向堆中相同的Deer對象。
## 虛方法
我們將介紹在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("Constructing an Employee");
this.name = name;
this.address = address;
this.number = number;
}
public void mailCheck()
{
System.out.println("Mailing a check to " + 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; //Annual salary
public Salary(String name, String address, int number, double
salary)
{
super(name, address, number);
setSalary(salary);
}
public void mailCheck()
{
System.out.println("Within mailCheck of Salary class ");
System.out.println("Mailing check to " + getName()
+ " with salary " + salary);
}
public double getSalary()
{
return salary;
}
public void setSalary(double newSalary)
{
if(newSalary >= 0.0)
{
salary = newSalary;
}
}
public double computePay()
{
System.out.println("Computing salary pay for " + getName());
return salary/52;
}
}
```
現在我們仔細閱讀下面的代碼,嘗試給出它的輸出結果:
```
/* 文件名 : VirtualDemo.java */
public class VirtualDemo
{
public static void main(String [] args)
{
Salary s = new Salary("Mohd Mohtashim", "Ambehta, UP", 3, 3600.00);
Employee e = new Salary("John Adams", "Boston, MA", 2, 2400.00);
System.out.println("Call mailCheck using Salary reference --");
s.mailCheck();
System.out.println("\n Call mailCheck using Employee reference--");
e.mailCheck();
}
}
```
以上實例編譯運行結果如下:
```
Constructing an Employee
Constructing an Employee
Call mailCheck using Salary reference --
Within mailCheck of Salary class
Mailing check to Mohd Mohtashim with salary 3600.0
Call mailCheck using Employee reference--
Within mailCheck of Salary class
Mailing check to John Adams with salary 2400.0
```
例子中,我們實例化了兩個Salary對象。一個使用Salary引用s,另一個使用Employee引用。
編譯時,編譯器檢查到mailCheck()方法在Salary類中的聲明。
在調用s.mailCheck()時,Java虛擬機(JVM)調用Salary類的mailCheck()方法。
因為e是Employee的引用,所以調用e的mailCheck()方法則有完全不同的結果。
當編譯器檢查e.mailCheck()方法時,編譯器檢查到Employee類中的mailCheck()方法。
在編譯的時候,編譯器使用Employee類中的mailCheck()方法驗證該語句, 但是在運行的時候,Java虛擬機(JVM)調用的是Salary類中的mailCheck()方法。
該行為被稱為虛擬方法調用,該方法被稱為虛擬方法。
Java中所有的方法都能以這種方式表現,借此,重寫的方法能在運行時調用,不管編譯的時候源代碼中引用變量是什么數據類型。
- Java 基礎
- Java 簡介
- Java開發環境配置
- Java基礎語法
- Java對象和類
- Java基本數據類型
- Java變量類型
- Java修飾符
- Java運算符
- Java循環結構 - for, while 及 do...while
- Java分支結構 - if...else/switch
- Java Number類
- Java Character類
- Java String類
- Java StringBuffer和StringBuilder類
- Java 數組
- Java 日期時間
- Java正則表達式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 異常處理
- Java 面向對象
- Java 繼承
- Java 重寫(Override)與重載(Overload)
- Java 多態
- Java 抽象類
- Java 接口
- Java 包(package)
- Java 高級教程
- Java 數據結構
- Java Enumeration接口
- Java Bitset類
- Java Vector 類
- Java Stack 類
- Java Dictionary 類
- Java Hashtable 接口
- Java Properties 接口
- Java 集合框架
- Java 泛型
- Java序列化
- Java 網絡編程
- Java 發送郵件
- Java 多線程編程
- Java Applet基礎
- Java 文檔注釋
- Servlet 教程
- Servlet 簡介
- Servlet 環境設置
- Servlet 生命周期
- Servlet 實例
- Servlet 表單數據
- Servlet 客戶端 HTTP 請求
- Servlet 服務器 HTTP 響應
- Servlet HTTP 狀態碼
- Servlet 編寫過濾器
- Servlet 異常處理
- Servlet Cookies 處理
- Servlet Session 跟蹤
- Servlet 數據庫訪問
- Servlet 文件上傳
- Servlet 處理日期
- Servlet 網頁重定向
- Servlet 點擊計數器
- Servlet 自動刷新頁面
- Servlet 發送電子郵件
- Servlet 包
- Servlet 調試
- Servlet 國際化
- JSP 基礎
- JSP 簡介
- JSP 開發環境搭建
- JSP 結構
- JSP 生命周期
- JSP 語法
- JSP 指令
- JSP 動作元素
- JSP 動作元素
- JSP 隱含對象
- JSP 客戶端請求
- JSP 服務器響應
- JSP HTTP 狀態碼
- JSP 表單處理
- JSP 過濾器
- JSP Cookies 處理
- JSP Session
- JSP 文件上傳
- JSP 日期處理
- JSP 頁面重定向
- JSP 點擊量統計
- JSP 自動刷新
- JSP 發送郵件
- JSP 高級教程
- JSP 標準標簽庫(JSTL)
- JSP 連接數據庫
- JSP XML 數據處理
- JSP JavaBean
- JSP 自定義標簽
- JSP 表達式語言
- JSP 異常處理
- JSP 調試
- JSP 國際化
- 免責聲明