Java接口(interface)的概念及使用

在抽象類中,能夠包含一個或多個抽象方法;但在接口(interface)中,全部的方法必須都是抽象的,不能有方法體,它比抽象類更加「抽象」。

接口使用 interface 關鍵字來聲明,能夠看作是一種特殊的抽象類,能夠指定一個類必須作什麼,而不是規定它如何去作。

現實中也有不少接口的實例,好比說天津關鍵詞優化公司串口電腦硬盤,Serial ATA委員會指定了Serial ATA 2.0規範,這種規範就是接口。Serial ATA委員會不負責生產硬盤,只是指定通用的規範。

希捷、日立、三星等生產廠家會按照規範生產符合接口的硬盤,這些硬盤就能夠實現通用化,若是正在用一塊160G日立的串口硬盤,如今要升級了,能夠購買一塊320G的希捷串口硬盤,安裝上去就能夠繼續使用了。

下面的代碼能夠模擬Serial ATA委員會定義如下串口硬盤接口:html

  1. //串行硬盤接口
  2. public interface SataHdd{
  3. //鏈接線的數量
  4. public static final int CONNECT_LINE=4;
  5. //寫數據
  6. public void writeData(String data);
  7. //讀數據
  8. public String readData();
  9. }

注意:接口中聲明的成員變量默認都是 public static final 的,必須顯示的初始化。於是在常量聲明時能夠省略這些修飾符。

接口是若干常量和抽象方法的集合,目前看來和抽象類差很少。確實如此,接口本就是從抽象類中演化而來的,於是除特別規定,接口享有和類一樣的「待遇」。好比,源程序中能夠定義多個類或接口,但最多隻能有一個public 的類或接口,若是有則源文件必須取和public的類和接口相同的名字。和類的繼承格式同樣,接口之間也能夠繼承,子接口能夠繼承父接口中的常量和抽象方法並添加新的抽象方法等。

但接口有其自身的一些特性,概括以下。

1) 接口中只能定義抽象方法,這些方法默認爲 public abstract 的,於是在聲明方法時能夠省略這些修飾符。試圖在接口中定義實例變量、非抽象的實例方法及靜態方法,都是非法的。例如:java

  1. public interface SataHdd{
  2. //鏈接線的數量
  3. public int connectLine; //編譯出錯,connectLine被看作靜態常量,必須顯式初始化
  4. //寫數據
  5. protected void writeData(String data); //編譯出錯,必須是public類型
  6. //讀數據
  7. public static String readData(){ //編譯出錯,接口中不能包含靜態方法
  8. return "數據"; //編譯出錯,接口中只能包含抽象方法,
  9. }
  10. }


3) 接口中沒有構造方法,不能被實例化。

4) 一個接口不實現另外一個接口,但能夠繼承多個其餘接口。接口的多繼承特色彌補了類的單繼承。例如:框架

  1. //串行硬盤接口
  2. public interface SataHdd extends A,B{
  3. // 鏈接線的數量
  4. public static final int CONNECT_LINE = 4;
  5. // 寫數據
  6. public void writeData(String data);
  7. // 讀數據
  8. public String readData();
  9. }
  10. interface A{
  11. public void a();
  12. }
  13. interface B{
  14. public void b();
  15. }

爲何使用接口

大型項目開發中,可能須要從繼承鏈的中間插入一個類,讓它的子類具有某些功能而不影響它們的父類。例如 A -> B -> C -> D -> E,A 是祖先類,若是須要爲C、D、E類添加某些通用的功能,最簡單的方法是讓C類再繼承另一個類。可是問題來了,Java 是一種單繼承的語言,不能再讓C繼承另一個父類了,只到移動到繼承鏈的最頂端,讓A再繼承一個父類。這樣一來,對C、D、E類的修改,影響到了整個繼承鏈,不具有可插入性的設計。

接口是可插入性的保證。在一個繼承鏈中的任何一個類均可以實現一個接口,這個接口會影響到此類的全部子類,但不會影響到此類的任何父類。此類將不得不實現這個接口所規定的方法,而子類能夠今後類自動繼承這些方法,這時候,這些子類具備了可插入性。

咱們關心的不是哪個具體的類,而是這個類是否實現了咱們須要的接口。

接口提供了關聯以及方法調用上的可插入性,軟件系統的規模越大,生命週期越長,接口使得軟件系統的靈活性和可擴展性,可插入性方面獲得保證。

接口在面向對象的 Java 程序設計中佔有舉足輕重的地位。事實上在設計階段最重要的任務之一就是設計出各部分的接口,而後經過接口的組合,造成程序的基本框架結構。優化

接口的使用

接口的使用與類的使用有些不一樣。在須要使用類的地方,會直接使用new關鍵字來構建一個類的實例,但接口不能夠這樣使用,由於接口不能直接使用 new 關鍵字來構建實例。

接口必須經過類來實現(implements)它的抽象方法,而後再實例化類。類實現接口的關鍵字爲implements。

若是一個類不能實現該接口的全部抽象方法,那麼這個類必須被定義爲抽象方法。

不容許建立接口的實例,但容許定義接口類型的引用變量,該變量指向了實現接口的類的實例。

一個類只能繼承一個父類,但卻能夠實現多個接口。

實現接口的格式以下:
修飾符 class 類名 extends 父類 implements 多個接口 {
    實現方法
}

請看下面的例子:url

  1. import static java.lang.System.*;
  2. public class Demo{
  3. public static void main(String[] args) {
  4. SataHdd sh1=new SeagateHdd(); //初始化希捷硬盤
  5. SataHdd sh2=new SamsungHdd(); //初始化三星硬盤
  6. }
  7. }
  8. //串行硬盤接口
  9. interface SataHdd{
  10. //鏈接線的數量
  11. public static final int CONNECT_LINE=4;
  12. //寫數據
  13. public void writeData(String data);
  14. //讀數據
  15. public String readData();
  16. }
  17. // 維修硬盤接口
  18. interface fixHdd{
  19. // 維修地址
  20. String address = "北京市海淀區";
  21. // 開始維修
  22. boolean doFix();
  23. }
  24. //希捷硬盤
  25. class SeagateHdd implements SataHdd, fixHdd{
  26. //希捷硬盤讀取數據
  27. public String readData(){
  28. return "數據";
  29. }
  30. //希捷硬盤寫入數據
  31. public void writeData(String data) {
  32. out.println("寫入成功");
  33. }
  34. // 維修希捷硬盤
  35. public boolean doFix(){
  36. return true;
  37. }
  38. }
  39. //三星硬盤
  40. class SamsungHdd implements SataHdd{
  41. //三星硬盤讀取數據
  42. public String readData(){
  43. return "數據";
  44. }
  45. //三星硬盤寫入數據
  46. public void writeData(String data){
  47. out.println("寫入成功");
  48. }
  49. }
  50. //某劣質硬盤,不能寫數據
  51. abstract class XXHdd implements SataHdd{
  52. //硬盤讀取數據
  53. public String readData() {
  54. return "數據";
  55. }
  56. }

接口做爲類型使用

接口做爲引用類型來使用,任何實現該接口的類的實例均可以存儲在該接口類型的變量中,經過這些變量能夠訪問類中所實現的接口中的方法,Java 運行時系統會動態地肯定應該使用哪一個類中的方法,其實是調用相應的實現類的方法。

示例以下:spa

  1. public class Demo{
  2. public void test1(A a) {
  3. a.doSth();
  4. }
  5. public static void main(String[] args) {
  6. Demo d = new Demo();
  7. A a = new B();
  8. d.test1(a);
  9. }
  10. }
  11. interface A {
  12. public int doSth();
  13. }
  14. class B implements A {
  15. public int doSth() {
  16. System.out.println("now in B");
  17. return 123;
  18. }
  19. }

運行結果:
now in B

你們看到接口能夠做爲一個類型來使用,把接口做爲方法的參數和返回類型。.net

相關文章
相關標籤/搜索