接口和抽象類的區別


    需求:接口是否可繼承接口? 抽象類是否可實現(implements)接口? 抽象類是否可繼承實體類(concrete class)? 抽象類中是否可以有靜態的main方法?

先說明二者的定義,然后聊聊需求,最后分析二者的區別。

    含有abstract修飾符的類即為抽象類,抽象類不能創建實例對象。含有抽象方法的類必須定義為abstract class。在abstract class中,方法不必是抽象的,但是抽象方法必須在具體子類中實現,所以,不能有抽象構造方法或抽象靜態方法。子類如果沒有實現抽象父類中的所有抽象方法,則必須定義為abstract類型。抽象類需要被繼承才能使用,而被final修飾的類無法被繼承,所以abstract和final是不能共存的。

    接口(interface)可以說成是抽象類的一種特例,接口中的所有方法都必須是抽象的。接口中的方法定義默認為public abstract類型,接口中的成員變量類型默認為public static final。

    接口可以繼承接口。

    抽象類可以實現(implements)接口。

    抽象類可以繼承實體類。但和實體類的繼承一樣,也要求父類可繼承,並且擁有子類可以訪問到的構造函數。其實Object就是個實體類,Java的API文檔里,每個抽象類的條目里都明確寫着直接或間接繼承自Object,所以這點是沒有疑問的。

    抽象類中可以有靜態的main方法。下面分析二者的區別。

    備注:只要明白了接口和抽象類的本質和作用,這些問題都很好回答,你想想,如果你是java語言的設計者,你是否會提供這樣的支持,如果不提供的話,有什么理由嗎?如果你沒有道理不提供,那答案就是肯定的了。

     只有記住抽象類與普通類的區別就是①不能創建實例對象,②允許有abstract方法。也可以這么理解——抽象類就是一個不能實例化的普通類,不過如果方法加了abstract,那么就必須在子類里面重寫。

 

抽象類

接口

方法默認實現

支持

不支持,接口完全是抽象的

實現

子類使用extends關鍵字來繼承抽象類。子類如果不是抽象類,需要實現抽象類中聲明的所有抽象方法

子類使用關鍵字implements來實現接口,需要實現接口中聲明的所有方法

是否有構造函數

與正常Java類的區別

不能實例化抽象類,因為有abstract方法

接口是完全不同的類型

訪問修飾符

publicprotecteddefault

只有public

main方法

支持

不支持

多繼承

繼承一個類和實現多個接口

只可以繼承一個或多個其它接口

速度

速度快

稍微有點慢,因為它需要時間去尋找在類中實現的方法

添加新方法

添加后可以給它提供默認的實現,故不需要改變現在的代碼

添加后必須改變實現該接口的類

    抽象類為什么不能實例化對象?

    現實生活中也有抽象類的例子,比如說人類是一個抽象類,無法創建一個稱作人類的對象,但是,人可以在繼承人類后來創建對象。況且抽象類中的抽象方法只有聲明,沒有主體,如果實例化了,又如何去實現調用呢?

    什么時候使用抽象類和接口?

•如果擁有一些方法並且想讓它們中的一些有默認實現,那么使用抽象類吧。

•如果想實現多重繼承,那么必須使用接口。由於Java不支持多繼承,子類不能夠繼承多個類,但可以實現多個接口。因此就可以使用接口來解決它。

•如果基本功能在不斷改變,那么就需要使用抽象類。如果不斷改變基本功能並且使用接口,那么就需要改變所有實現了該接口的類。

    下面接着再說說兩者在應用上的區別。接口更多的是在系統架構設計方法發揮作用,主要用於定義模塊之間的通信契約。而抽象類在代碼實現方面發揮作用,可以實現代碼的重用,例如,模板方法設計模式是抽象類的一個典型應用,假設某個項目的所有HTTP請求都要用相同的方式進行權限判斷、訪問日志記錄和異常處理,那么就可以定義一個抽象的基類,讓所有的controller都繼承這個抽象基類,在抽象基類的service方法中實現上述功能,在各個子類中只是完成各自的業務邏輯代碼,偽代碼如下:

import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; 

public abstract class BaseServlet extends HttpServlet {

  public final void service(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
   // 記錄訪問日志
   // 進行權限判斷
  } 
   protected abstract void doService(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException;
   // 注意訪問權限定義成protected,顯得既專業,又嚴謹,因為它是專門給子類用的
} 

class MyServlet1 extends BaseServlet {
   protected void doService(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
   // 本Servlet只處理的具體業務邏輯代碼
   }
}

    父類方法中間的某段代碼不確定,留給子類干,就用模板方法設計模式。

    備注:這道題的思路是先從總體解釋抽象類和接口的基本概念,然后再比較兩者的語法細節,最后再說兩者的應用區別。比較兩者語法細節區別的條理是:先從一個類中的構造方法、普通成員變量和方法(包括抽象方法),靜態變量和方法,繼承性等6個方面逐一去比較回答,接着從第三者繼承的角度的回答,特別是最后用了一個典型的例子來展現自己深厚的技術功底。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM