java中extends與implements的區(qū)別淺談
1. 在類的聲明中,通過關(guān)鍵字extends來創(chuàng)建一個(gè)類的子類。一個(gè)類通過關(guān)鍵字implements聲明自己使用一個(gè)或者多個(gè)接口。
extends 是繼承某個(gè)類, 繼承之后可以使用父類的方法, 也可以重寫父類的方法; implements 是實(shí)現(xiàn)多個(gè)接口, 接口的方法一般為空的, 必須重寫才能使用
2.extends是繼承父類,只要那個(gè)類不是聲明為final或者那個(gè)類定義為abstract的就能繼承,JAVA中不支持多重繼承,但是可以用接口 來實(shí)現(xiàn),這樣就要用到implements,繼承只能繼承一個(gè)類,但implements可以實(shí)現(xiàn)多個(gè)接口,用逗號(hào)分開就行了
比如
class A extends B implements C,D,E
學(xué)了好久,今天終于明白了implements,其實(shí)很簡(jiǎn)單,看下面幾個(gè)例子就ok啦~~
接口的一些概念
<!--[if !supportLineBreakNewLine]-->
public inerface Runner
{
int ID = 1;
void run ();
}
interface Animal extends Runner
{
void breathe ();
}
class Fish implements Animal
{
public void run ()
{
System.out.println("fish is swimming");
}
public void breather()
{
System.out.println("fish is bubbing");
}
}
abstract LandAnimal implements Animal
{
public void breather ()
{
System.out.println("LandAnimal is breathing");
}
}
class Student extends Person implements Runner
{
......
public void run ()
{
System.out.println("the student is running");
}
......
}
interface Flyer
{
void fly ();
}
class Bird implements Runner , Flyer
{
public void run ()
{
System.out.println("the bird is running");
}
public void fly ()
{
System.out.println("the bird is flying");
}
}
class TestFish
{
public static void main (String args[])
{
Fish f = new Fish();
int j = 0;
j = Runner.ID;
j = f.ID;
}
}
接口實(shí)現(xiàn)的注意點(diǎn):
a.實(shí)現(xiàn)一個(gè)接口就是要實(shí)現(xiàn)該接口的所有的方法(抽象類除外)。
b.接口中的方法都是抽象的。
c.多個(gè)無關(guān)的類可以實(shí)現(xiàn)同一個(gè)接口,一個(gè)類可以實(shí)現(xiàn)多個(gè)無關(guān)的接口。
===========================================================
extends與implements的不同
extends是繼承父類,只要那個(gè)類不是聲明為final或者那個(gè)類定義為abstract的就能繼承,JAVA中不支持多重 繼承,但是可以用接口來實(shí)現(xiàn),這樣就要用到implements,繼承只能繼承一個(gè)類,但implements可以實(shí)現(xiàn)多個(gè)接口,用逗號(hào)分開就行了
比如
class A extends B implements C,D,E
一個(gè)類通過關(guān)鍵字implements聲明自己使用一個(gè)或者多個(gè)接口。在類的聲明中,通過關(guān)鍵字extends來創(chuàng)建一個(gè)類的子類。
class 子類名 extends 父類名 implenments 接口名
{...
}
===========================================================
A a = new B(); 結(jié)果a是一個(gè)A類的實(shí)例,只能訪問A中的方法,那么又和A a = new A();有什么區(qū)別呢?
==========================================================
class B extends A
繼承過后通常會(huì)定義一些父類沒有的成員或者方法。
A a = new B();
這樣是可以的,上傳。
a是一個(gè)父類對(duì)象的實(shí)例,因而不能訪問子類定義的新成員或方法。
==========================================================
假如這樣定義:
class A{
int i;
void f(){}
}
class B extends A{
int j;
void f(){}//重寫
void g(){}
}
然后:
B b = new B();
b就是子類對(duì)象的實(shí)例,不僅能夠訪問自己的屬性和方法,也能夠訪問父類的屬性和方法。諸如b.i,b.j,b.f(),b.g()都是合法的。此時(shí) b.f()是訪問的B中的f()
A a = new B();
a雖然是用的B的構(gòu)造函數(shù),但經(jīng)過upcast,成為父類對(duì)象的實(shí)例,不能訪問子類的屬性和方法。a.i,a.f()是合法的,而a.j,a.g()非 法。此時(shí)訪問a.f()是訪問B中的f()
==========================================================
A a = new B(); 這條語句,實(shí)際上有三個(gè)過程:
(1) A a;
將a聲明為父類對(duì)象,只是一個(gè)引用,未分配空間
(2) B temp = new B();
通過B類的構(gòu)造函數(shù)建立了一個(gè)B類對(duì)象的實(shí)例,也就是初始化
(3) a = (A)temp;
將子類對(duì)象temp轉(zhuǎn)換未父類對(duì)象并賦給a,這就是上傳(upcast),是安全的。
經(jīng)過以上3個(gè)過程,a就徹底成為了一個(gè)A類的實(shí)例。
子類往往比父類有更多的屬性和方法,上傳只是舍棄,是安全的;而下傳(downcast)有時(shí)會(huì)增加,通常是不安全的。
===========================================================
a.f()對(duì)應(yīng)的應(yīng)該是B類的方法f()
調(diào)用構(gòu)造函數(shù)建立實(shí)例過后,對(duì)應(yīng)方法的入口已經(jīng)確定了。
如此以來,a雖被上傳為A類,但其中重寫的方法f()仍然是B的方法f()。也就是說,每個(gè)對(duì)象知道自己應(yīng)該調(diào)用哪個(gè)方法。
A a1 = new B();
A a2 = new C();
a1,a2兩個(gè)雖然都是A類對(duì)象,但各自的f()不同。這正是1樓說的多態(tài)性的體現(xiàn)。
這類問題在《Java編程思想》上都講的很清楚
implements
implements是一個(gè)類實(shí)現(xiàn)一個(gè)接口用的關(guān)鍵字,他是用來實(shí)現(xiàn)接口中定義的抽象方法。比如:people是一個(gè)接口,他里面有say這個(gè)方法。public interface people(){ public say();}但是接口沒有方法體。只能通過一個(gè)具體的類去實(shí)現(xiàn)其中的方法體。比如chinese這個(gè)類,就實(shí)現(xiàn)了people這個(gè)接口。 public class chinese implements people{ public say() {System.out.println("你好!");}}
在java中implements表示子類繼承父類,如類A繼承類B寫成 class A implements B{}
與Extends的不同
extends, 可以實(shí)現(xiàn)父類,也可以調(diào)用父類初始化 this.parent()。而且會(huì)覆蓋父類定義的變量或者函數(shù)。這樣的好處是:架構(gòu)師定義好接口,讓工程師實(shí)現(xiàn)就可以了。整個(gè)項(xiàng)目開發(fā)效率和開發(fā)成本大大降低。
implements,實(shí)現(xiàn)父類,子類不可以覆蓋父類的方法或者變量。即使子類定義與父類相同的變量或者函數(shù),也會(huì)被父類取代掉。
這兩種實(shí)現(xiàn)的具體使用,是要看項(xiàng)目的實(shí)際情況,需要實(shí)現(xiàn),不可以修改implements,只定義接口需要具體實(shí)現(xiàn),或者可以被修改擴(kuò)展性好,用extends。
<!--[endif]-->
相關(guān)文章
hibernate 配置數(shù)據(jù)庫(kù)方言的實(shí)現(xiàn)方法
這篇文章主要介紹了hibernate 配置數(shù)據(jù)庫(kù)方言的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05SpringBoot默認(rèn)包掃描機(jī)制及@ComponentScan指定掃描路徑詳解
這篇文章主要介紹了SpringBoot默認(rèn)包掃描機(jī)制及@ComponentScan指定掃描路徑詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11關(guān)于IDEA配置Hibernate中遇到的問題解決
這篇文章主要給大家介紹了關(guān)于IDEA配置Hibernate中遇到的問題,文中通過圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05解決使用json-lib包實(shí)現(xiàn)xml轉(zhuǎn)json時(shí)空值被轉(zhuǎn)為空中括號(hào)的問題
網(wǎng)上能查到的xml轉(zhuǎn)json的jar包大部分是net.sf.json-lib,但是JSON json =xmlSerializer.read(xml); 方法會(huì)出現(xiàn)將空值轉(zhuǎn)化為[]的問題,下面為大家提供兩種解決方法2018-03-03詳解使用Spring Boot開發(fā)Restful程序
本篇文章主要介紹了詳解使用Spring Boot開發(fā)Restful程序,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05Java實(shí)現(xiàn)Huffman編碼的示例代碼
Huffman編碼是一種編碼方式,本文主要介紹了Java實(shí)現(xiàn)Huffman編碼的示例代碼,具有一定的參考價(jià)值,感興趣的可以了解一下2023-08-08JavaEE多線程中阻塞隊(duì)列的項(xiàng)目實(shí)踐
本文主要介紹了JavaEE多線程中阻塞隊(duì)列的項(xiàng)目實(shí)踐,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-09-09