溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

[轉(zhuǎn)載]Java面向?qū)ο蟪绦蛟O(shè)計(jì)之接口應(yīng)用

發(fā)布時(shí)間:2020-08-11 14:04:25 來(lái)源:ITPUB博客 閱讀:118 作者:ccainiao1 欄目:編程語(yǔ)言
Java語(yǔ)言提供了一種接口(interface)機(jī)制。這種接口機(jī)制使Java的面向?qū)ο缶幊套兊酶屿`活。我們可以用接口來(lái)定義一個(gè)類的表現(xiàn)形式,但接口不能包含任何實(shí)現(xiàn)。在《Thinking in Java》一書中,作者對(duì)接口有這樣的描述:“接口(interface)比抽象(abstract)的概念更進(jìn)了一步。你可以把一個(gè)接口看成是一個(gè)純的抽象類。”我認(rèn)為作者對(duì)接口的這一解釋再準(zhǔn)確不過(guò)了。
理解并用好接口機(jī)制將幫助我們更好的掌握J(rèn)ava這種面向?qū)ο蟮木幊陶Z(yǔ)言。下面我們來(lái)討論一下接口的使用規(guī)則以及相關(guān)的應(yīng)用。[@more@]

一、接口的定義及實(shí)現(xiàn)

   定義接口和定義類相似,只是要把 class關(guān)鍵字換為 interface。定義方法時(shí)只需要方法名,返回類型和參數(shù)列表,不能有方法體。接口中可以定義字段,這些字段都被暗指為 static 和 final,因此應(yīng)該根據(jù)需要先定好這些字段的值。例如:

public interface Flyable {
void fly();
}

public interface Talkable {
void talk();
}

public interface Message {
int MAX_SIZE = 4096;
String getMessage();
}

  上面定義的幾個(gè)接口中,F(xiàn)lyable 和 Talkable 只定義了一個(gè)方法,而 Message 里除了方法外還有一個(gè)字段 MAX_SIZE??梢钥闯鲞@些接口只定義了類的表現(xiàn)形式,而不包含任何實(shí)現(xiàn),所以不能直接使用。要使用這些接口就需要有相應(yīng)的類去實(shí)現(xiàn)它們。實(shí)現(xiàn)接口時(shí)應(yīng)該先在類名后用 implements 關(guān)鍵字申明將要實(shí)現(xiàn)的接口,如果要實(shí)現(xiàn)多個(gè)接口,應(yīng)該用逗號(hào)將它們隔開,然后一一實(shí)現(xiàn)這些接口中定義的方法。如下面的例子:

public class Parrot implements Flyable, Talkable {

public void fly() {
System.out.println("Flying like a parrot…");
}

public void talk() {
System.out.println("Hello! I am a parrot!");
}
}

public class TextMessage implements Message {
String message;

public void setMessage(String msg) {
message = msg;
if (message.length() > MAX_SIZE)
message = message.substring(0, MAX_SIZE);
}

public String getMessage() {
return message;
}
}

  在 Parrot(鸚鵡)例子中,我們用接口 Flyable 來(lái)表示飛行能力,Talkable 表示說(shuō)話能力,但它們并不包含具體實(shí)現(xiàn)。而 Parrot 同時(shí)具有這兩種能力,所以我們?yōu)?Parrot 類同時(shí)實(shí)現(xiàn)了 Flyable 和 Talkable 這兩個(gè)接口。同樣我們還可以定義一個(gè)Swallow(燕子)類,但燕子只有飛行能力,所以我們只需要為 Swallow 實(shí)現(xiàn) Flyable 就行了。因?yàn)樗鼈兏髯缘牡娘w行方法有所不同,所以它們有各自關(guān)于飛行的具體實(shí)現(xiàn)。

  另外,正因?yàn)橐粋€(gè)類可以同時(shí)實(shí)現(xiàn)多個(gè)接口,使得Java的面向?qū)ο筇匦宰兊梅浅l`活。運(yùn)用這種特性,我們可以實(shí)現(xiàn)類似C++語(yǔ)言中多繼承那樣的特性,甚至更靈活的一些特性。下面我們來(lái)討論一下接口在實(shí)際中的應(yīng)用。

二、用接口來(lái)定義一些全局變量

   因?yàn)榻涌趦?nèi)的字段都是static和final的,所以我們可以很方便的利用這一點(diǎn)來(lái)創(chuàng)建一些常量。例如:

public interface Constants {
String ROOT = "/root";
int MAX_COUNT = 200;
int MIN_COUNT = 100;
}

  在使用時(shí)可以直接用Constants.ROOT這樣的形式來(lái)引用其中的常量。我們還可以用下面這種方法來(lái)創(chuàng)建初始值不確定的常量。

public interface RandomColor {
int red = Math.random() * 255;
int green = Math.random() * 255;
int blue = Math.random() * 255;
}

  其中red、green和blue的值會(huì)在第一次被訪問(wèn)時(shí)建立,然后保持不變。

三、用接口來(lái)定義基本數(shù)據(jù)結(jié)構(gòu)
  在設(shè)計(jì)一套軟件系統(tǒng)的初期,我們可以用接口來(lái)對(duì)一些基本數(shù)據(jù)元素的特性來(lái)進(jìn)行一些描述,再根據(jù)需要進(jìn)行不同的實(shí)現(xiàn)。請(qǐng)大家看看下面這個(gè)例子:

public interface User {
int getAge();
String getName();
String getPassword();
}

public class XMLUser implements User {
// 這里用XML技術(shù)實(shí)現(xiàn)User接口中的方法
public int getAge() { ... }
public String getName() { ... }
public String getPassword() { ... }
}

public abstract class UserFactory {
public static UserFactory getUserFactory() {
return new XMLUserFactory();
}

public User getUser(String name);
public User getAdmin();
public User createUser(String name, String password, int age);
public void addUser(User user);
public void delUser(User user);
}

public class XMLUserFactory extends UserFactory {
// 這里用XML技術(shù)實(shí)現(xiàn)的UserFactory的抽象方法
}

  在這個(gè)例子中,我們定義了一個(gè)接口User和一個(gè)抽象類UserFactory。然后我們用XML技術(shù)實(shí)現(xiàn)這兩個(gè)類。可以看出,我們只需要從用UserFactory的getUserFactory()就可以得到一個(gè)UserFactory的實(shí)例,而不用去考慮這個(gè)實(shí)例的具體實(shí)現(xiàn)方法。通過(guò)UserFactory的這個(gè)實(shí)例我們還可以直接得到User的實(shí)例,也不用去考具體的實(shí)現(xiàn)方法。

  如果我們決定用JDBC技術(shù)來(lái)實(shí)現(xiàn)User和UserFactory,我們只需要按上面的形式實(shí)現(xiàn)JDBCUser和JDBCUserFactory就行了。然后把UserFactory中的getUserFactory方法修改一下就可以改變了它們的實(shí)現(xiàn)方法。而我們已經(jīng)寫好的調(diào)用UserFactory和User的部分不需要做任何修改。

  這是用接口來(lái)定義數(shù)據(jù)結(jié)構(gòu)的一個(gè)簡(jiǎn)單的例子,在實(shí)際應(yīng)用中還有很多靈活的使用方法,大家需要在學(xué)習(xí)過(guò)程中不斷的去體會(huì)。

四、理解分布式應(yīng)用的原理

  目前有很多軟件項(xiàng)目都使用了分布式的技術(shù)。Java 有多種支持分布式應(yīng)用的技術(shù),早期用的比較多的有 RMI、CORBA 等技術(shù),而現(xiàn)在 EJB 技術(shù)更為流行一些。但這些技術(shù)不管怎么發(fā)展,其實(shí)都是以接口為基礎(chǔ)的。

  以遠(yuǎn)程方法調(diào)用 RMI(Remote Method Invocation)為例。在編寫 RMI 應(yīng)用時(shí),我們需要做兩件最基本的事,首先要定義一個(gè)接口,這個(gè)接口要繼承 java.rmi.Remote 接口,這個(gè)接口中應(yīng)該包含你要從遠(yuǎn)端調(diào)用的方法名。接下來(lái)就是寫一個(gè)類來(lái)實(shí)現(xiàn)這個(gè)接口中的方法。例如:

public interface Product extends java.rmi.Remote {
String getName() throws java.rmi.RemoteException;
}

public class ProductImpl implements Product {
String name;

public ProductImpl(String n) {
name = n;
}

public String getName() throws java.rmi.RemoteException {
return name;
}
}

  在這個(gè)例子中,接口 Product 是放在客戶端的,而 ProductImpl 是放在服務(wù)器端的,客戶在使用時(shí)只需要用指定的規(guī)則得到Product 的實(shí)例就行了,不用去考慮 Product 接口里的方法是如何實(shí)現(xiàn)的。在定義好這兩個(gè)類后,用Java開發(fā)包命令“rmic ProductImpl”就可以幫助我們自動(dòng)生成兩個(gè)類 ProductImpl_Skel 和 ProductImpl_Stub。這兩個(gè)類就包含了RMI調(diào)用的運(yùn)作機(jī)制。有興趣的朋友可以把這兩個(gè)類反編譯后研究一下。你會(huì)發(fā)現(xiàn)其中 ProductImpl_Stub 實(shí)際上是接口 Product 的一個(gè)實(shí)現(xiàn)類。RMI 機(jī)制就是用這個(gè)類來(lái)生成 Product 的實(shí)例供客戶端使用。另一個(gè)類 ProductImpl_Skel 則是在服務(wù)端響應(yīng) ProductImpl_Stub 的調(diào)用請(qǐng)求的類。而 RMI 最底層的通訊原理則是利用 ObjectInputStream 和 ObjetOutputStream 通過(guò) Socket 將要調(diào)用的方法名及參數(shù)列表傳到服務(wù)器端,服務(wù)器端再通過(guò)特定的方法調(diào)用實(shí)現(xiàn)類(在本例中是 ProductImpl)的對(duì)應(yīng)方法,然后將結(jié)果通過(guò) Socket 傳回客戶端就行了。由于 Skel 和 Stub 類是用工具生成的,所以就大大節(jié)省了開發(fā)的時(shí)間。另外,如果我們需要修改一些實(shí)現(xiàn)方法或錯(cuò)誤,只需要對(duì)服務(wù)器端的實(shí)現(xiàn)類進(jìn)行修改就可以了,也就是說(shuō)這種分布式應(yīng)用的大部分維護(hù)工作在服務(wù)器端就可以完成。

   現(xiàn)在越來(lái)越多的應(yīng)用使用了 EJB 這種技術(shù)。EJB 是從 RMI 發(fā)展而來(lái)的一項(xiàng)技術(shù),它比RMI定義得更加完善,可以獲得更好的面向?qū)ο蟮奶匦?。但它的?guī)則要比RMI復(fù)雜一些。但是不管它多復(fù)雜,它同樣是使用了接口來(lái)定義各種不同的 Bean,也同樣需要編寫相應(yīng)的實(shí)現(xiàn)類來(lái)完成具體的功能,最后還要通過(guò) Socket 來(lái)進(jìn)行通訊。EJB的運(yùn)作機(jī)制本身有一定的復(fù)雜性,所以其應(yīng)用的效率理所當(dāng)然就會(huì)受到一定的影響。因此在選擇開發(fā)技術(shù)時(shí)應(yīng)該根據(jù)應(yīng)用的規(guī)模和特點(diǎn)仔細(xì)考慮,不一定流行的技術(shù)就一定能適應(yīng)你的應(yīng)用。如果你很好的掌握了面向?qū)ο蟮脑O(shè)計(jì)原則,你就可以自行設(shè)計(jì)。也許你可以根據(jù)自己應(yīng)用的特點(diǎn)設(shè)計(jì)出更合適的分布式應(yīng)用結(jié)構(gòu)。

五、結(jié)論

   除了上述的一些應(yīng)用外,還有很多地方可以使用接口,比如在Java的事件機(jī)制中就常用到接口。另外,對(duì)于一些已經(jīng)開發(fā)好的系統(tǒng),在結(jié)構(gòu)上進(jìn)行較大的調(diào)整已經(jīng)不太現(xiàn)實(shí),這時(shí)可以通過(guò)定義一些接口并追加相應(yīng)的實(shí)現(xiàn)來(lái)完成功能結(jié)構(gòu)的擴(kuò)展。

   總之,學(xué)好接口可以幫助我們更好的理解和運(yùn)用面向?qū)ο蟮脑O(shè)計(jì)原則。使我們能設(shè)計(jì)出更好的軟件系統(tǒng)。由于本人水平的限制,如有錯(cuò)誤之處還請(qǐng)多多指正。謝謝!

參考文獻(xiàn)
1. 《Thinking in Java》2nd Edition
2. 《Enterprise JavaBeans》3rd Edition
3. 《Java2核心》第二版

-------

文章出處:http://softbbs.pconline.com.cn/topic.jsp?tid=6093250&topicPage=3

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI