在自上而下的繼承層次結(jié)構(gòu)中,位于上層的類更具有通用性,甚至可能更加抽象。從某種角度看,祖先類更加通用,它只包含一些最基本的成員,人們只將它作為派生其他類的基類,而不會用來創(chuàng)建對象。甚至,你可以只給出方法的定義而不實現(xiàn),由子類根據(jù)具體需求來具體實現(xiàn)。
這種只給出方法定義而不具體實現(xiàn)的方法被稱為抽象方法,抽象方法是沒有方法體的,在代碼的表達(dá)上就是沒有“{}”。包含一個或多個抽象方法的類也必須被聲明為抽象類。
使用 abstract 修飾符來表示抽象方法和抽象類。
抽象類除了包含抽象方法外,還可以包含具體的變量和具體的方法。類即使不包含抽象方法,也可以被聲明為抽象類,防止被實例化。
抽象類不能被實例化,抽象方法必須在子類中被實現(xiàn)。請看下面的代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
import static java.lang.System.*; public final class Demo{ public static void main(String[] args) { Teacher t = new Teacher(); t.setName( "王明" ); t.work(); Driver d = new Driver(); d.setName( "小陳" ); d.work(); } } // 定義一個抽象類 abstract class People{ private String name; // 實例變量 // 共有的 setter 和 getter 方法 public void setName(String name){ this .name = name; } public String getName(){ return this .name; } // 抽象方法 public abstract void work(); } class Teacher extends People{ // 必須實現(xiàn)該方法 public void work(){ out.println( "我的名字叫" + this .getName() + ",我正在講課,請大家不要東張西望..." ); } } class Driver extends People{ // 必須實現(xiàn)該方法 public void work(){ out.println( "我的名字叫" + this .getName() + ",我正在開車,不能接聽電話..." ); } } |
運行結(jié)果:
1
2
|
我的名字叫王明,我正在講課,請大家不要東張西望... 我的名字叫小陳,我正在開車,不能接聽電話... |
關(guān)于抽象類的幾點說明:
抽象類不能直接使用,必須用子類去實現(xiàn)抽象類,然后使用其子類的實例。然而可以創(chuàng)建一個變量,其類型是一個抽象類,并讓它指向具體子類的一個實例,也就是可以使用抽象類來充當(dāng)形參,實際實現(xiàn)類作為實參,也就是多態(tài)的應(yīng)用。
不能有抽象構(gòu)造方法或抽象靜態(tài)方法。
在下列情況下,一個類將成為抽象類:
當(dāng)一個類的一個或多個方法是抽象方法時;
當(dāng)類是一個抽象類的子類,并且不能為任何抽象方法提供任何實現(xiàn)細(xì)節(jié)或方法主體時;
當(dāng)一個類實現(xiàn)一個接口,并且不能為任何抽象方法提供實現(xiàn)細(xì)節(jié)或方法主體時;注意:
這里說的是這些情況下一個類將成為抽象類,沒有說抽象類一定會有這些情況。
一個典型的錯誤:抽象類一定包含抽象方法。 但是反過來說“包含抽象方法的類一定是抽象類”就是正確的。
事實上,抽象類可以是一個完全正常實現(xiàn)的類