Java 中extends(類別繼承)與implements(介面實作)使用方法

Java 中extends(類別繼承)與implements(介面實作)使用方法

Java 中extends(類別繼承)與implements(介面實作)使用方法

 

資料來源: http://blog.csdn.net/chen_chun_guang/article/details/6323201

 

初學Java語言, 代碼中的extendsimplements讓我感到很迷惑,現在終於弄明白它們之間的區別和用法了。

//定義一個Runner介面

public inerface Runner

{

   int ID = 1;

   void run ();

}

 

//定義一個interface Animal,它繼承于父類Runner

interface Animal extends Runner

{

   void breathe ();

}

 

//定義Fish類,它實現了Animal介面的方法run()和breather()

class Fish implements Animal

{

   public void run ()    //實現了Animal方法run()

 {

    System.out.println(“fish is swimming”);

 }

public void breather()

 {

    System.out.println(“fish is bubbing”);  

 }

}

//定義了一個抽象類別LandAnimal,它實現了介面Animal的方法。

abstract LandAnimal implements Animal

{

 

   public void breather ()

 {

    System.out.println(“LandAnimal is breathing”);

 }

}

//定義了一個類Student,它繼承了類Person,並實現了Runner介面的方法run()。

class Student extends Person implements Runner

{

    ……

    public void run ()

     {

          System.out.println(“the student is running”);

     }

    ……

}

 

//定義了一個介面Flyer

interface Flyer

{

   void fly ();

}

 

//定義了一個類Bird,它實現了Runner和Flyer這兩個介面定義的方法。

class Bird implements Runner , Flyer

{

   public void run ()   //Runner介面定義的方法。

    {

        System.out.println(“the bird is running”);

    }

   public void fly ()   //Flyer介面定義的方法。

    {

        System.out.println(“the bird is flying”);

    }

}

 

//TestFish類

class TestFish

{

   public static void main (String args[])

    {

       Fish f = new Fish();

       int j = 0;

       j = Runner.ID;

       j = f.ID;

    }

}

 

介面實現的注意點:

a)實現一個介面就是要實現該介面的所有的方法(抽象類別除外)
b)介面中的方法都是抽象的。
c)多個無關的類可以實現同一個介面,一個類可以實現多個無關的介面。

 

extends與implements的區別:

extends 是繼承父類,只要那個類不是聲明為final或者那個類定義為abstract的就能繼承,JAVA中不支持多重繼承,但是可以用介面來實現,這樣就用到了implements,繼承只能繼承一個類,但implements可以實現多個介面,用逗號分開就行了。

比如: 

class A extends B implements C,D,E {}    class 子類名 extends 父類名 implenments 介面名)

 

父類與子類繼承關係上的不同:

A a = new B(); 結果a是一個A類的實例,只能訪問A中的方法,那麼又和A a = new A();有什麼區別呢?

***********************************************************************************************

class B extends A
繼承過後通常會定義一些父類沒有的成員或者方法。
A a = new B();
這樣是可以的,上傳。
a是一個父類物件的實例,因而不能訪問子類定義的新成員或方法。

***********************************************************************************************

假如這樣定義:
class A

{
   int i;
   void f(){}
}
class B extends A

{
    int j;
    void f(){}       //
重寫
    void g(){}
}

然後:
B b = new B();
b
就是子類物件的實例,不僅能夠訪問自己的屬性和方法,也能夠訪問父類的屬性和方法。諸如b.i,b.j,b.f(),b.g()都是合法的。此時b.f()是訪問的B中的f()

A a = new B();
a
雖然是用的B的構造函數,但經過upcast,成為父類物件的實例,不能訪問子類的屬性和方法。a.i,a.f()是合法的,而a.j,a.g()非法。此時訪問a.f()是訪問B中的f()

***********************************************************************************************

A a = new B(); 這條語句,實際上有三個過程:
(1) A a;
a聲明為父類物件,只是一個引用,未配置空間
(2) B temp = new B();
通過B類的構造函數建立了一個B類物件的實例,也就是初始化
(3) a = (A)temp;
將子類物件temp轉換未父類物件並賦給a,這就是上傳(upcast),是安全的。
經過以上3個過程,a就徹底成為了一個A類的實例。
子類往往比父類有更多的屬性和方法,上傳只是捨棄,是安全的;而下傳(downcast)有時會增加,通常是不安全的。

***********************************************************************************************

a.f()對應的應該是B類的方法f()
調用構造函數建立實例過後,對應方法的入口已經確定了。
如此以來,a雖被上傳為A類,但其中重寫的方法f()仍然是B的方法f()。也就是說,每個物件知道自己應該調用哪個方法。
A a1 = new B();
A a2 = new C();
a1,a2
兩個雖然都是A類物件,但各自的f()不同。這正是多態性的體現。

***********************************************************************************************

 


 


發表迴響

你的電子郵件位址並不會被公開。 必要欄位標記為 *