Java基礎(chǔ)學(xué)習(xí)之接口詳解
概述
接口,是Java語言中一種引用類型,是方法的集合,如果說類的內(nèi)部封裝了成員變量、構(gòu)造方法和成員方法,那么接口的內(nèi)部主要就是封裝了方法,包含抽象方法(JDK 7及以前),默認(rèn)方法和靜態(tài)方法(JDK 8),私有方法 (JDK 9)。 接口的定義,它與定義類方式相似,但是使用 interface 關(guān)鍵字。它也會被編譯成.class文件,但一定要明確它并 不是類,而是另外一種引用數(shù)據(jù)類型。
引用數(shù)據(jù)類型:數(shù)組,類,接口。
接口的使用,它不能創(chuàng)建對象,但是可以被實(shí)現(xiàn)( implements ,類似于被繼承)。一個(gè)實(shí)現(xiàn)接口的類(可以看做是接口的子類),需要實(shí)現(xiàn)接口中所有的抽象方法,創(chuàng)建該類對象,就可以調(diào)用方法了,否則它必須是一個(gè)抽象類。
定義格式
public interface 接口名稱 { // 抽象方法 // 默認(rèn)方法 // 靜態(tài)方法 // 私有方法 }
含有抽象方法
抽象方法:使用abstract 關(guān)鍵字修飾,可以省略,沒有方法體。該方法供子類實(shí)現(xiàn)使用。 代碼如下:
public interface InterFaceName { public abstract void method(); }
含有默認(rèn)方法和靜態(tài)方法
默認(rèn)方法:使用 default 修飾,不可省略,供子類調(diào)用或者子類重寫。 靜態(tài)方法:使用 static 修飾,供接口直接調(diào)用。 代碼如下:
public interface InterFaceName { public default void method() { // 執(zhí)行語句 } public static void method2() { // 執(zhí)行語句 } }
含有私有方法和私有靜態(tài)方法
私有方法:使用 private 修飾,供接口中的默認(rèn)方法或者靜態(tài)方法調(diào)用。 代碼如下:
public interface InterFaceName { private void method() { // 執(zhí)行語句 } }
基本的實(shí)現(xiàn)
實(shí)現(xiàn)的概述
類與接口的關(guān)系為實(shí)現(xiàn)關(guān)系,即類實(shí)現(xiàn)接口,該類可以稱為接口的實(shí)現(xiàn)類,也可以稱為接口的子類。實(shí)現(xiàn)的動(dòng)作類似繼承,格式相仿,只是關(guān)鍵字不同,實(shí)現(xiàn)使用 implements 關(guān)鍵字。 非抽象子類實(shí)現(xiàn)接口:
- 必須重寫接口中所有抽象方法。
- 繼承了接口的默認(rèn)方法,即可以直接調(diào)用,也可以重寫。 實(shí)現(xiàn)格式:
class 類名 implements 接口名 { // 重寫接口中抽象方法【必須】 // 重寫接口中默認(rèn)方法【可選】 }
抽象方法的使用
必須全部實(shí)現(xiàn),代碼如下: 定義接口:
public interface LiveAble { // 定義抽象方法 public abstract void eat(); public abstract void sleep(); }
定義實(shí)現(xiàn)類:
public class Animal implements LiveAble { @Override public void eat() { System.out.println("吃東西"); } @Override public void sleep() { System.out.println("睡覺"); } }
定義測試類:
public class InterfaceDemo { public static void main(String[] args) { // 創(chuàng)建子類對象 Animal a = new Animal(); // 調(diào)用實(shí)現(xiàn)后的方法 a.eat(); a.sleep(); } }
默認(rèn)方法的使用
可以繼承,可以重寫,二選一,但是只能通過實(shí)現(xiàn)類的對象來調(diào)用。
1.繼承默認(rèn)方法,代碼如下:
定義接口:
public interface LiveAble { public default void fly(){ System.out.println("天上飛"); } }
定義實(shí)現(xiàn)類:
public class Animal implements LiveAble { // 繼承,什么都不用寫,直接調(diào)用 }
定義測試類:
public class InterfaceDemo { public static void main(String[] args) { // 創(chuàng)建子類對象 Animal a = new Animal(); // 調(diào)用默認(rèn)方法 a.fly(); } }
輸出結(jié)果:
天上飛
2.重寫默認(rèn)方法,代碼如下:
定義接口:
public interface LiveAble { public default void fly(){ System.out.println("天上飛"); } }
定義實(shí)現(xiàn)類:
public class Animal implements LiveAble { @Override public void fly() { System.out.println("自由自在的飛"); } }
定義測試類:
public class InterfaceDemo { public static void main(String[] args) { // 創(chuàng)建子類對象 Animal a = new Animal(); // 調(diào)用默認(rèn)方法 a.fly(); } }
輸出結(jié)果:
自由自在的飛
靜態(tài)方法的使用
靜態(tài)與.class 文件相關(guān),只能使用接口名調(diào)用,不可以通過實(shí)現(xiàn)類的類名或者實(shí)現(xiàn)類的對象調(diào)用,代碼如下: 定義接口:
public interface LiveAble { public static void run(){ System.out.println("跑起來~~~"); } }
定義實(shí)現(xiàn)類:
public class Animal implements LiveAble { // 無法重寫靜態(tài)方法 }
定義測試類:
public class InterfaceDemo { public static void main(String[] args) { // Animal.run(); // 【錯(cuò)誤】無法繼承方法,也無法調(diào)用 LiveAble.run(); // } }
輸出結(jié)果:
跑起來~~~
私有方法的使用
私有方法:只有默認(rèn)方法可以調(diào)用。 私有靜態(tài)方法:默認(rèn)方法和靜態(tài)方法可以調(diào)用。 如果一個(gè)接口中有多個(gè)默認(rèn)方法,并且方法中有重復(fù)的內(nèi)容,那么可以抽取出來,封裝到私有方法中,供默認(rèn)方法去調(diào)用。從設(shè)計(jì)的角度講,私有的方法是對默認(rèn)方法和靜態(tài)方法的輔助。同學(xué)們在已學(xué)技術(shù)的基礎(chǔ)上,可以自行測試。 定義接口:
public interface LiveAble { default void func(){ func1(); func2(); } private void func1(){ System.out.println("跑起來~~~"); } private void func2(){ System.out.println("跑起來~~~"); } }
接口的多實(shí)現(xiàn)
在繼承體系中,一個(gè)類只能繼承一個(gè)父類。而對于接口而言,一個(gè)類是可以實(shí)現(xiàn)多個(gè)接口的,這叫做接口的多實(shí)現(xiàn)。并且,一個(gè)類能繼承一個(gè)父類,同時(shí)實(shí)現(xiàn)多個(gè)接口。 實(shí)現(xiàn)格式:
class 類名 [extends 父類名] implements 接口名1,接口名2,接口名3... { // 重寫接口中抽象方法【必須】 // 重寫接口中默認(rèn)方法【不重名時(shí)可選】 }
[ ]: 表示可選操作。
抽象方法
接口中,有多個(gè)抽象方法時(shí),實(shí)現(xiàn)類必須重寫所有抽象方法。如果抽象方法有重名的,只需要重寫一次。代碼如下: 定義多個(gè)接口:
interface A { public abstract void showA(); public abstract void show(); } interface B { public abstract void showB(); public abstract void show(); }
定義實(shí)現(xiàn)類:
public class C implements A,B{ @Override public void showA() { System.out.println("showA"); } @Override public void showB() { System.out.println("showB"); } @Override public void show() { System.out.println("show"); } }
默認(rèn)方法
接口中,有多個(gè)默認(rèn)方法時(shí),實(shí)現(xiàn)類都可繼承使用。如果默認(rèn)方法有重名的,必須重寫一次。代碼如下: 定義多個(gè)接口:
interface A { public default void methodA(){} public default void method(){} } interface B { public default void methodB(){} public default void method(){} }
定義實(shí)現(xiàn)類:
public class C implements A,B{ @Override public void method() { System.out.println("method"); } }
靜態(tài)方法
接口中,存在同名的靜態(tài)方法并不會沖突,原因是只能通過各自接口名訪問靜態(tài)方法。
優(yōu)先級的問題
當(dāng)一個(gè)類,既繼承一個(gè)父類,又實(shí)現(xiàn)若干個(gè)接口時(shí),父類中的成員方法與接口中的默認(rèn)方法重名,子類就近選擇執(zhí)行父類的成員方法。代碼如下: 定義接口:
interface A { public default void methodA(){ System.out.println("AAAAAAAAAAAA"); } }
定義父類:
class D { public void methodA(){ System.out.println("DDDDDDDDDDDD"); } }
定義子類:
class C extends D implements A { // 未重寫methodA方法 }
定義測試類:
public class Test { public static void main(String[] args) { C c = new C(); c.methodA(); } }
輸出結(jié)果:
DDDDDDDDDDDD
接口的多繼承
一個(gè)接口能繼承另一個(gè)或者多個(gè)接口,這和類之間的繼承比較相似。接口的繼承使用 extends 關(guān)鍵字,子接口繼 承父接口的方法。如果父接口中的默認(rèn)方法有重名的,那么子接口需要重寫一次。代碼如下: 定義父接口:
interface A { public default void method(){ System.out.println("AAAAAAAAAAAAAAAAAAA"); } } interface B { public default void method(){ System.out.println("BBBBBBBBBBBBBBBBBBB"); } }
定義子接口:
interface D extends A,B{ @Override public default void method() { System.out.println("DDDDDDDDDDDDDD"); } }
小貼士: 子接口重寫默認(rèn)方法時(shí),default關(guān)鍵字可以保留。 子類重寫默認(rèn)方法時(shí),default關(guān)鍵字不可以保留。
其他成員特點(diǎn)
接口中,無法定義成員變量,但是可以定義常量,其值不可以改變,默認(rèn)使用public static final修飾。 接口中,沒有構(gòu)造方法,不能創(chuàng)建對象。 接口中,沒有靜態(tài)代碼塊。
到此這篇關(guān)于Java基礎(chǔ)學(xué)習(xí)之接口詳解的文章就介紹到這了,更多相關(guān)Java接口內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決Springboot中Feignclient調(diào)用時(shí)版本問題
這篇文章主要介紹了解決Springboot中Feign?client調(diào)用時(shí)版本問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03Apache Calcite進(jìn)行SQL解析(java代碼實(shí)例)
Calcite是一款開源SQL解析工具, 可以將各種SQL語句解析成抽象語法樹AST(Abstract Syntax Tree), 之后通過操作AST就可以把SQL中所要表達(dá)的算法與關(guān)系體現(xiàn)在具體代碼之中,今天通過代碼實(shí)例給大家介紹Apache Calcite進(jìn)行SQL解析問題,感興趣的朋友一起看看吧2022-01-01Java讀取網(wǎng)絡(luò)文件的實(shí)例代碼
這篇文章主要介紹了Java讀取網(wǎng)絡(luò)文件的實(shí)例代碼,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07Java8實(shí)現(xiàn)Stream流的合并的方法展示
本文介紹了Java8中Stream流的合并方法,包括concat()、flatMap()和reduce()三種方法。其中,concat()方法可以將兩個(gè)Stream流合并成一個(gè),flatMap()方法可以將一個(gè)Stream流中的元素映射成多個(gè)Stream流并合并成一個(gè),reduce()方法可以將Stream流中的元素逐個(gè)合并成一個(gè)結(jié)果2023-05-05SpringBoot與velocity的結(jié)合的示例代碼
本篇文章主要介紹了SpringBoot與velocity的結(jié)合的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-03-03