· 爲什麼需要接口?接口和抽象類的區別?
接口就是比“抽象類”還“抽象”的“抽象類”,可以更加規範的對子類進行約束。全面地專業地實現了:規範和具體實現的分離。
抽象類還提供某些具體實現,接口不提供任何實現,接口中所有方法都是抽象方法。接口是完全面向規範的,規定了一批類具有的公共方法規範。
從接口的實現者角度看,接口定義了可以向外部提供的服務。
從接口的調用者角度看,接口定義了實現者能提供那些服務。
接口是兩個模塊之間通信的標準,通信的規範。如果能把你要設計的模塊之間的接口定義好,就相當於完成了系統的設計大綱,剩下的就是添磚加瓦的具體實現了。大家在工作以後,做系統時往往就是使用“面向接口”的思想來設計系統。
接口和實現類不是父子關係,是實現規則的關係。比如:我定義一個接口Runnable,Car實現它就能在地上跑,Train實現它也能在地上跑,飛機實現它也能在地上跑。就是說,如果它是交通工具,就一定能跑,但是一定要實現Runnable接口。
· 接口的本質探討
接口就是規範,定義的是一組規則,體現了現實世界中“如果你是…則必須能…”的思想。如果你是天使,則必須能飛。如果你是汽車,則必須能跑。如果你是好人,則必須能幹掉壞人;如果你是壞人,則必須欺負好人。
接口的本質是契約,就像我們人間的法律一樣。制定好後大家都遵守。
面向對象的精髓,是對對象的抽象,最能體現這一點的就是接口。爲什麼我們討論設計模式都只針對具備了抽象能力的語言(比如C++、Java、C#等),就是因爲設計模式所研究的,實際上就是如何合理的去抽象。
區別
1. 普通類:具體實現
2. 抽象類:具體實現,規範(抽象方法)
3. 接口:規範!
聲明格式:
[訪問修飾符]
interface
接口名 [
extends
父接口
1
,父接口
2
…] {
常量定義;
方法定義;
}
定義接口的詳細說明:
1. 訪問修飾符:只能是public或默認。
2. 接口名:和類名採用相同命名機制。
3. extends:接口可以多繼承。
4. 常量:接口中的屬性只能是常量,總是:public static final 修飾。不寫也是。
5. 方法:接口中的方法只能是:public abstract。 省略的話,也是public abstract。
要點
1. 子類通過implements來實現接口中的規範。
2. 接口不能創建實例,但是可用於聲明引用變量類型。
3. 一個類實現了接口,必須實現接口中所有的方法,並且這些方法只能是public的。
4. JDK1.7之前,接口中只能包含靜態常量、抽象方法,不能有普通屬性、構造方法、普通方法。
5. JDK1.8後,接口中包含普通的靜態方法。
【示例5-17】接口的使用
public
class
TestInterface {
public
static
void
main(String[] args) {
Volant volant =
new
Angel();
volant.fly();
System.out.println(Volant.FLY_HIGHT);
Honest honest =
new
GoodMan();
honest.helpOther();
}
}
/**飛行接口*/
interface
Volant {
int
FLY_HIGHT =
100
;
// 總是:public static final類型的;
void
fly();
//總是:public abstract void fly();
}
/**善良接口*/
interface
Honest {
void
helpOther();
}
/**Angle類實現飛行接口和善良接口*/
class
Angel
implements
Volant, Honest{
public
void
fly() {
System.out.println(
"我是天使,飛起來啦!"
);
}
public
void
helpOther() {
System.out.println(
"扶老奶奶過馬路!"
);
}
}
class
GoodMan
implements
Honest {
public
void
helpOther() {
System.out.println(
"扶老奶奶過馬路!"
);
}
}
class
BirdMan
implements
Volant {
public
void
fly() {
System.out.println(
"我是鳥人,正在飛!"
);
}
}
執行結果如果5-23所示:
圖5-23 示例5-17運行效果圖
接口完全支持多繼承。和類的繼承類似,子接口擴展某個父接口,將會獲得父接口中所定義的一切。
【示例5-18】接口的多繼承
interface
A {
void
testa();
}
interface
B {
void
testb();
}
/**接口可以多繼承:接口C繼承接口A和B*/
interface
C
extends
A, B {
void
testc();
}
public
class
Test
implements
C {
public
void
testc() {
}
public
void
testa() {
}
public
void
testb() {
}
}