Java接口和Java抽象類

Java接口和Java抽象類有太多相似的地方,又有太多特別的地方,究竟在什麼地方,纔是它們的最佳位置呢?把它們比較一下,你就可以發現了。

1、Java接口和Java抽象類最大的一個區別,就在於Java抽象類可以提供某些方法的部分實現,而Java接口不可以,這大概就是Java抽象類唯一的優點吧,但這個優點非常有用。
如果向一個抽象類里加入一個新的具體方法時,那麼它所有的子類都一下子都得到了這個新方法,而Java接口做不到這一點,如果向一個Java接口裏加入一個新方法,所有實現這個接口的類就無法成功通過編譯了,因爲你必須讓每一個類都再實現這個方法才行,這顯然是Java接口的缺點。

2、一個抽象類的實現只能由這個抽象類的子類給出,也就是說,這個實現處在抽象類所定義出的繼承的等級結構中,而由於Java語言的單繼承性,所以抽象類作爲類型定義工具的效能大打折扣。
在這一點上,Java接口的優勢就出來了,任何一個實現了一個Java接口所規定的方法的類都可以具有這個接口的類型,而一個類可以實現任意多個Java接口,從而這個類就有了多種類型。

3、從第2點不難看出,Java接口是定義混合類型的理想工具,混合類表明一個類不僅僅具有某個主類型的行爲,而且具有其他的次要行爲。

4、結合1、2點中抽象類和Java接口的各自優勢,具精典的設計模式就出來了:聲明類型的工作仍然由Java接口承擔,但是同時給出一個Java抽象類,且實現了這個接口,而其他同屬於這個抽象類型的具體類可以選擇實現這個Java接口,也可以選擇繼承這個抽象類,也就是說在層次結構中,Java接口在最上面,然後緊跟着抽象類,哈,這下兩個的最大優點都能發揮到極至了。這個模式就是“缺省適配模式”。
在Java語言API中用了這種模式,而且全都遵循一定的命名規範:Abstract +接口名。

Java接口和Java抽象類的存在就是爲了用於具體類的實現和繼承的,如果你準備寫一個具體類去繼承另一個具體類的話,那你的設計就有很大問題了。Java抽象類就是爲了繼承而存在的,它的抽象方法就是爲了強制子類必須去實現的。

使用Java接口和抽象Java類進行變量的類型聲明、參數是類型聲明、方法的返還類型說明,以及數據類型的轉換等。而不要用具體Java類進行變量的類型聲明、參數是類型聲明、方法的返還類型說明,以及數據類型的轉換等。

我想,如果你編的代碼裏面連一個接口和抽象類都沒有的話,也許我可以說你根本沒有用到任何設計模式,任何一個設計模式都是和抽象分不開的,而抽象與Java接口和抽象Java類又是分不開的。

理解抽象,理解Java接口和抽象Java類,我想就應該是真正開始用面向對象的思想去分析問題,解決問題了吧。

接口和抽象類的不同點有:  

(1)抽象類可以有實例變量,而接口不能擁有實例變量,接口中的變量都是靜態(static)的常量(final)。
  
(2)抽象類可以有非抽象方法,而接口只能有抽象方法。
  
java中,類與類之間是不能多繼承的。java之所以禁止類與類之間的多繼承是因爲多繼承有很大的缺點。

多繼承雖然能使子類同時擁有多個父類的特徵,但是其缺點也是很顯著的,主要有兩方面:

(1)如果在一個子類繼承的多個父類中擁有相同名字的實例變量,子類在引用該變量時將產生歧義,無法判斷應該使用哪個父類的變量。例如:
  
類ClassA:
public class ClassA {
  protected int varSame = 0 ;
  }

  類ClassB:
  public class ClassB {
  protected int varSame = 1 ;
  }

  子類ClassC:(假設允許類與類之間多繼承) 
 public class ClassC extends ClassA, ClassB {
  public void printOut() {
  System.out.println( super .varSame);
  }
  public static void main(String[] args) {
  ClassC classC = new ClassC();
  classC.printOut();
  }
  }

  上面程序的運行結果會是什麼呢?輸出0還是1?
  
(2)如果在一個子類繼承的多個父類中擁有相同方法,子類中有沒有覆蓋該方法,那麼調用該方法時將產生歧義,無法判斷應該調用哪個父類的方法。例如:
  類ClassA:  
public class ClassA {
  public void printOut() {
  System.out.println( 0 );
  }
  }

  類ClassB:  
public class ClassB {
  public void printOut() {
  System.out.println( 1 );
  }
  }

  子類ClassC:(假設允許類與類之間多繼承)
public class ClassC extends ClassA, ClassB {
  public static void main(String[] args) {
  ClassA classA = new ClassC();
  classA.printOut(); // ------------------------- A行
  ClassB classB = new ClassC();
  classB.printOut(); // ------------------------- B行
  ClassC classC = new ClassC();
  classC.printOut(); //------------------------- C行
  }
  }

上面程序的運行結果會是什麼呢?A、B、C三行的輸出是0還是1?


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章