標籤:
很多常見的面試題都會出諸如抽象類別和介面有什麼區別,什麼情況下會使用抽象類別和什麼情況你會使用介面這樣的問題。本文我們將仔細討論這些話題。
在討論它們之間的不同點之前,我們先看看抽象類別、介面各自的特性。
抽象類別
抽象類別是用來捕捉子類的通用特性的 。它不能被執行個體化,只能被用作子類的超類。抽象類別是被用來建立繼承層級裡子類的模板。以JDK中的GenericServlet為例:
public abstract class GenericServlet implements Servlet, ServletConfig, Serializable { // abstract method abstract void service(ServletRequest req, ServletResponse res); void init() { // Its implementation } // other method related to Servlet}
當HttpServlet類繼承GenericServlet時,它提供了service方法的實現:
public class HttpServlet extends GenericServlet { void service(ServletRequest req, ServletResponse res) { // implementation } protected void doGet(HttpServletRequest req, HttpServletResponse resp) { // Implementation } protected void doPost(HttpServletRequest req, HttpServletResponse resp) { // Implementation } // some other methods related to HttpServlet}
介面
介面是抽象方法的集合。如果一個類實現了某個介面,那麼它就繼承了這個介面的抽象方法。這就像契約模式,如果實現了這個介面,那麼就必須確保使用這些方法。介面只是一種形式,介面自身不能做任何事情。以Externalizable介面為例:
public interface Externalizable extends Serializable { void writeExternal(ObjectOutput out) throws IOException; void readExternal(ObjectInput in) throws IOException, ClassNotFoundException;}
當你實現這個介面時,你就需要實現上面的兩個方法:
public class Employee implements Externalizable { int employeeId; String employeeName; @Override public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { employeeId = in.readInt(); employeeName = (String) in.readObject(); } @Override public void writeExternal(ObjectOutput out) throws IOException { out.writeInt(employeeId); out.writeObject(employeeName); }}
抽象類別和介面的對比
參數 |
抽象類別 |
介面 |
預設的方法實現 |
它可以有預設的方法實現 |
介面完全是抽象的。它根本不存在方法的實現 |
實現 |
子類使用extends關鍵字來繼承抽象類別。如果子類不是抽象類別的話,它需要提供抽象類別中所有聲明的方法的實現。 |
子類使用關鍵字implements來實現介面。它需要提供介面中所有聲明的方法的實現 |
構造器 |
抽象類別可以有構造器 |
介面不能有構造器 |
與正常Java類的區別 |
除了你不能執行個體化抽象類別之外,它和普通Java類沒有任何區別 |
介面是完全不同的類型 |
存取修飾詞 |
抽象方法可以有public、protected和default這些修飾符 |
介面方法預設修飾符是public。你不可以使用其它修飾符。 |
main方法 |
抽象方法可以有main方法並且我們可以運行它 |
介面沒有main方法,因此我們不能運行它。 |
多繼承 |
抽象方法可以繼承一個類和實現多個介面 |
介面只可以繼承一個或多個其它介面 |
速度 |
它比介面速度要快 |
介面是稍微有點慢的,因為它需要時間去尋找在類中實現的方法。 |
添加新方法 |
如果你往抽象類別中添加新的方法,你可以給它提供預設的實現。因此你不需要改變你現在的代碼。 |
如果你往介面中添加方法,那麼你必須改變實現該介面的類。
|
什麼時候使用抽象類別和介面????
如果你擁有一些方法並且想讓它們中的一些有預設實現,那麼使用抽象類別吧。
如果你想實現多重繼承,那麼你必須使用介面。由於Java不支援多繼承,子類不能夠繼承多個類,但可以實現多個介面。因此你就可以使用介面來解決它。
如果準系統在不斷改變,那麼就需要使用抽象類別。如果不斷改變準系統並且使用介面,那麼就需要改變所有實現了該介面的類。
Java抽象類別與介面的區別