摘要:如果一個(gè)非抽象類遵循了某個(gè)接口,就必須實(shí)現(xiàn)該接口中的所有方法。抽象類是對(duì)整個(gè)類整體進(jìn)行抽象,包括屬性行為,但是接口卻是對(duì)類局部行為進(jìn)行抽象。因此最好的解決辦法是多帶帶將報(bào)警設(shè)計(jì)為一個(gè)接口,包含行為設(shè)計(jì)為多帶帶的一個(gè)抽象類,包含和兩種行為。
在了解抽象類之前,先來(lái)了解一下抽象方法。抽象方法是一種特殊的方法:它只有聲明,而沒有具體的實(shí)現(xiàn)。抽象方法的聲明格式為:
abstract void fun();
抽象方法必須用abstract關(guān)鍵字進(jìn)行修飾。如果一個(gè)類含有抽象方法,則稱這個(gè)類為抽象類,抽象類必須在類前用abstract關(guān)鍵字修飾。因?yàn)槌橄箢愔泻袩o(wú)具體實(shí)現(xiàn)的方法,所以不能用抽象類創(chuàng)建對(duì)象。
下面要注意一個(gè)問題:在《JAVA編程思想》一書中,將抽象類定義為“包含抽象方法的類”,但是后面發(fā)現(xiàn)如果一個(gè)類不包含抽象方法,只是用abstract修飾的話也是抽象類。也就是說(shuō)抽象類不一定必須含有抽象方法。個(gè)人覺得這個(gè)屬于鉆牛角尖的問題吧,因?yàn)槿绻粋€(gè)抽象類不包含任何抽象方法,為何還要設(shè)計(jì)為抽象類?所以暫且記住這個(gè)概念吧,不必去深究為什么。
public abstract class ClassName { abstract void fun(); }
從這里可以看出,抽象類就是為了繼承而存在的,如果你定義了一個(gè)抽象類,卻不去繼承它,那么等于白白創(chuàng)建了這個(gè)抽象類,因?yàn)槟悴荒苡盟鼇?lái)做任何事情。對(duì)于一個(gè)父類,如果它的某個(gè)方法在父類中實(shí)現(xiàn)出來(lái)沒有任何意義,必須根據(jù)子類的實(shí)際需求來(lái)進(jìn)行不同的實(shí)現(xiàn),那么就可以將這個(gè)方法聲明為abstract方法,此時(shí)這個(gè)類也就成為abstract類了。
包含抽象方法的類稱為抽象類,但并不意味著抽象類中只能有抽象方法,它和普通類一樣,同樣可以擁有成員變量和普通的成員方法。
抽象類和普通類的主要有三點(diǎn)區(qū)別:
1)抽象方法必須為public或者protected(因?yàn)槿绻麨閜rivate,則不能被子類繼承,子類便無(wú)法實(shí)現(xiàn)該方法),缺省情況下默認(rèn)為public。二.接口2)抽象類不能用來(lái)創(chuàng)建對(duì)象;
3)如果一個(gè)類繼承于一個(gè)抽象類,則子類必須實(shí)現(xiàn)父類的抽象方法。如果子類沒有實(shí)現(xiàn)父類的抽象方法,則必須將子類也定義為為abstract類。
在其他方面,抽象類和普通的類并沒有區(qū)別。
接口,英文稱作interface,在軟件工程中,接口泛指供別人調(diào)用的方法或者函數(shù)。從這里,我們可以體會(huì)到Java語(yǔ)言設(shè)計(jì)者的初衷,它是對(duì)行為的抽象。
在Java中,定一個(gè)接口的形式如下:
public interface InterfaceName { }
接口中可以含有變量和方法。但是要注意,接口中的變量會(huì)被隱式地指定為public static final變量(并且只能是public static final變量,用private修飾會(huì)報(bào)編譯錯(cuò)誤),而方法會(huì)被隱式地指定為public abstract方法且只能是public abstract方法(用其他關(guān)鍵字,比如private、protected、static、 final等修飾會(huì)報(bào)編譯錯(cuò)誤),并且接口中所有的方法不能有具體的實(shí)現(xiàn),也就是說(shuō),接口中的方法必須都是抽象方法。從這里可以隱約看出接口和抽象類的區(qū)別,接口是一種極度抽象的類型,它比抽象類更加“抽象”,并且一般情況下不在接口中定義變量。
要讓一個(gè)類遵循某組特地的接口需要使用implements關(guān)鍵字,具體格式如下:
class ClassName implements Interface1,Interface2,[....]{ }
可以看出,允許一個(gè)類遵循多個(gè)特定的接口。如果一個(gè)非抽象類遵循了某個(gè)接口,就必須實(shí)現(xiàn)該接口中的所有方法。對(duì)于遵循某個(gè)接口的抽象類,可以不實(shí)現(xiàn)該接口中的抽象方法。三.抽象類和接口的區(qū)別
1)抽象類可以提供成員方法的實(shí)現(xiàn)細(xì)節(jié),而接口中只能存在public abstract 方法;2)抽象類中的成員變量可以是各種類型的,而接口中的成員變量只能是public static final類型的;
3)接口中不能含有靜態(tài)代碼塊以及靜態(tài)方法,而抽象類可以有靜態(tài)代碼塊和靜態(tài)方法;
4)一個(gè)類只能繼承一個(gè)抽象類,而一個(gè)類卻可以實(shí)現(xiàn)多個(gè)接口。
1)抽象類是對(duì)一種事物的抽象,即對(duì)類抽象,而接口是對(duì)行為的抽象。抽象類是對(duì)整個(gè)類整體進(jìn)行抽象,包括屬性、行為,但是接口卻是對(duì)類局部(行為)進(jìn)行抽象。
舉個(gè)簡(jiǎn)單的例子,飛機(jī)和鳥是不同類的事物,但是它們都有一個(gè)共性,就是都會(huì)飛。那么在設(shè)計(jì)的時(shí)候,可以將飛機(jī)設(shè)計(jì)為一個(gè)類Airplane,將鳥設(shè)計(jì)為一個(gè)類Bird,但是不能將 飛行 這個(gè)特性也設(shè)計(jì)為類,因此它只是一個(gè)行為特性,并不是對(duì)一類事物的抽象描述。此時(shí)可以將 飛行 設(shè)計(jì)為一個(gè)接口Fly,包含方法fly( ),然后Airplane和Bird分別根據(jù)自己的需要實(shí)現(xiàn)Fly這個(gè)接口。然后至于有不同種類的飛機(jī),比如戰(zhàn)斗機(jī)、民用飛機(jī)等直接繼承Airplane即可,對(duì)于鳥也是類似的,不同種類的鳥直接繼承Bird類即可。從這里可以看出,繼承是一個(gè) "是不是"的關(guān)系,而 接口 實(shí)現(xiàn)則是 "有沒有"的關(guān)系。如果一個(gè)類繼承了某個(gè)抽象類,則子類必定是抽象類的種類,而接口實(shí)現(xiàn)則是有沒有、具備不具備的關(guān)系,比如鳥是否能飛(或者是否具備飛行這個(gè)特點(diǎn)),能飛行則可以實(shí)現(xiàn)這個(gè)接口,不能飛行就不實(shí)現(xiàn)這個(gè)接口。
2)設(shè)計(jì)層面不同,抽象類作為很多子類的父類,它是一種模板式設(shè)計(jì)。而接口是一種行為規(guī)范,它是一種輻射式設(shè)計(jì)。
什么是模板式設(shè)計(jì)?最簡(jiǎn)單例子,大家都用過ppt里面的模板,如果用模板A設(shè)計(jì)了ppt B和ppt C,ppt B和ppt C公共的部分就是模板A了,如果它們的公共部分需要改動(dòng),則只需要改動(dòng)模板A就可以了,不需要重新對(duì)ppt B和ppt C進(jìn)行改動(dòng)。而輻射式設(shè)計(jì),比如某個(gè)電梯都裝了某種報(bào)警器,一旦要更新報(bào)警器,就必須全部更新。也就是說(shuō)對(duì)于抽象類,如果需要添加新的方法,可以直接在抽象類中添加具體的實(shí)現(xiàn),子類可以不進(jìn)行變更;而對(duì)于接口則不行,如果接口進(jìn)行了變更,則所有實(shí)現(xiàn)這個(gè)接口的類都必須進(jìn)行相應(yīng)的改動(dòng)。
>下面看一個(gè)網(wǎng)上流傳最廣泛的例子:門和警報(bào)的例子:門都有open( )和close( )兩個(gè)動(dòng)作,此時(shí)我們可以定義通過抽象類和接口來(lái)定義這個(gè)抽象概念:
abstract class Door { public abstract void open(); public abstract void close(); }
或者:
interface Door { public abstract void open(); public abstract void close(); }
但是現(xiàn)在如果我們需要門具有報(bào)警alarm( )的功能,那么該如何實(shí)現(xiàn)?下面提供兩種思路:
> 1)將這三個(gè)功能都放在抽象類里面,但是這樣一來(lái)所有繼承于這個(gè)抽象類的子類都具備了報(bào)警功能,但是有的門并不一定具備報(bào)警功能;
> 2)將這三個(gè)功能都放在接口里面,需要用到報(bào)警功能的類就需要實(shí)現(xiàn)這個(gè)接口中的open( )和close(),也許這個(gè)類根本就不具備open( )和close( )這兩個(gè)功能,比如火災(zāi)報(bào)警器。
從這里可以看出, Door的open()、close()和alarm()根本就屬于兩個(gè)不同范疇內(nèi)的行為,open()和close()屬于門本身固有的行為特性,而alarm()屬于延伸的附加行為。因此最好的解決辦法是多帶帶將報(bào)警設(shè)計(jì)為一個(gè)接口,包含alarm()行為,Door設(shè)計(jì)為多帶帶的一個(gè)抽象類,包含open和close兩種行為。再設(shè)計(jì)一個(gè)報(bào)警門繼承Door類和實(shí)現(xiàn)Alarm接口。
void alarm(); } abstract class Door { void open(); void close(); } class AlarmDoor extends Door implements Alarm { void oepn() { //.... } void close() { //.... } void alarm() { //.... } }
以線程池的例子來(lái)說(shuō)明接口、抽象類以及類在繼承和現(xiàn)實(shí)時(shí)需要注意的事項(xiàng)
1.接口繼承接口(接口不能實(shí)現(xiàn)接口)
Executor是線程池的頂層接口,只有一個(gè)方法
public interface Executor { void execute(Runnable command); }
public interface ExecutorService extends Executor { // ExecutorService 接口繼承自接口Executor void shutdown(); boolean isShutdown(); boolean isTerminated(); boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException;Future submit(Callable task); }
2.抽象類實(shí)現(xiàn)接口(抽象類不能繼承接口)
public abstract class AbstractExecutorService implements ExecutorService { // AbstractExecutorService 實(shí)現(xiàn)ExecutorService接口,不需要實(shí)現(xiàn)ExecutorService接口的方法,因?yàn)榻涌诳梢钥醋鲆环N特殊的抽象類 // 抽象類里邊的普通方法必須包含方法體,即必須有{},而抽象方法不能有方法體,必須用abstract聲明。 protectedRunnableFuture newTaskFor(Runnable runnable, T value) { }; protected RunnableFuture newTaskFor(Callable callable) { }; public Future> submit(Runnable task) {}; public Future submit(Runnable task, T result) { }; public Future submit(Callable task) { }; private T doInvokeAny(Collection extends Callable > tasks, boolean timed, long nanos) throws InterruptedException, ExecutionException, TimeoutException { }; } 3.普通類繼承抽象類或?qū)崿F(xiàn)接口
普通類繼承抽象類:必須實(shí)現(xiàn)抽象類的所有抽象方法和該抽象類父類的所有方法 普通類實(shí)現(xiàn)接口:必須實(shí)現(xiàn)接口及其父接口的全部方法public interface A{ void a(); } public interface B extends A{ void b(); } public abstract class C implements B{ void c(){}; abstract void c1(); } class D extends C{ @Override public void b() { // TODO Auto-generated method stub } @Override public void a() { // TODO Auto-generated method stub } @Override void c1() { // TODO Auto-generated method stub } } class E implements B{ @Override public void a() { // TODO Auto-generated method stub } @Override public void b() { // TODO Auto-generated method stub } }請(qǐng)尊重作者勞動(dòng)成果,轉(zhuǎn)載請(qǐng)標(biāo)明原文鏈接http://www.cnblogs.com/dolphi...
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/68591.html
摘要:抽象類和接口小記抽象類和接口實(shí)現(xiàn)了的多態(tài)多態(tài)是面向?qū)ο蟪绦蛘Z(yǔ)言的核心在項(xiàng)目開發(fā)過程中其實(shí)很少使用抽象類接口用得比較多今天小記一下抽象類和接口的區(qū)別抽象類抽象類不能被實(shí)例化抽象類可以繼承可以定義變量可以定義構(gòu)造方法抽象方法的要顯式的寫出來(lái)其子 Java抽象類和接口小記 Java抽象類和接口實(shí)現(xiàn)了java的多態(tài).多態(tài)是面向?qū)ο蟪绦蛘Z(yǔ)言的核心,在項(xiàng)目開發(fā)過程中,其實(shí)很少使用抽象類,接口用得比...
摘要:接口的對(duì)象可以利用子類對(duì)象的向上轉(zhuǎn)型進(jìn)行實(shí)例化賦值。接口文件保存在結(jié)尾的文件中,文件名使用接口名。接口相應(yīng)的字節(jié)碼文件必須在與包名稱相匹配的目錄結(jié)構(gòu)中。接口不能包含成員變量,除了全局常量定義。 概念 接口,在JAVA編程語(yǔ)言中是一個(gè)引用類型,是抽象方法的集合,接口通常以interface來(lái)聲明。一個(gè)類通過繼承接口的方式,從而來(lái)繼承接口的抽象方法。 接口中只能包含抽象方法和全局常量。 接...
摘要:最近一直在刷校招的題目,突然發(fā)現(xiàn)自己對(duì)接口和抽象類的掌握并不牢固,所以想記錄一下對(duì)接口和抽象類的理解一抽象類抽象方法是一種特殊的方法它只有聲明,而沒有具體的實(shí)現(xiàn)。因?yàn)槌橄箢愔泻袩o(wú)具體實(shí)現(xiàn)的方法,所以不能用抽象類創(chuàng)建對(duì)象。 最近一直在刷校招的題目,突然發(fā)現(xiàn)自己對(duì)接口和抽象類的掌握并不牢固,所以想記錄一下對(duì)java接口和抽象類的理解 一:抽象類 抽象方法是一種特殊的方法:它只有聲明,而沒...
摘要:很多常見的面試題都會(huì)出諸如抽象類和接口有什么區(qū)別,什么情況下會(huì)使用抽象類和什么情況你會(huì)使用接口這樣的問題。在討論它們之間的不同點(diǎn)之前,我們先看看抽象類接口各自的特性。抽象類抽象類是用來(lái)捕捉子類的通用特性的。 很多常見的面試題都會(huì)出諸如抽象類和接口有什么區(qū)別,什么情況下會(huì)使用抽象類和什么情況你會(huì)使用接口這樣的問題。本文我們將仔細(xì)討論這些話題。 在討論它們之間的不同點(diǎn)之前,我們先看看抽象類...
摘要:抽象類和抽象方法抽象方法這種方法是不完整的,僅有聲明而沒有方法。創(chuàng)建抽象類和抽象方法非常有用,因?yàn)樗麄兛梢允估鄣某橄笮悦鞔_起來(lái),并告訴用戶和編譯器打算怎樣來(lái)使用它們。接口用于建立類于類之間的協(xié)議。與抽象類相同,防止客戶端程序員創(chuàng)建該類對(duì)象。 點(diǎn)擊進(jìn)入我的博客 接口和內(nèi)部類為我們提供了一種將接口與實(shí)現(xiàn)分離的更加結(jié)構(gòu)化的方法。 9.1抽象類和抽象方法 抽象方法:這種方法是不完整的,僅有...
閱讀 1336·2021-10-27 14:14
閱讀 3587·2021-09-29 09:34
閱讀 2492·2019-08-30 15:44
閱讀 1735·2019-08-29 17:13
閱讀 2583·2019-08-29 13:07
閱讀 883·2019-08-26 18:26
閱讀 3353·2019-08-26 13:44
閱讀 3220·2019-08-26 13:37