中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中怎么實現分布式計算

發布時間:2021-07-01 17:25:01 來源:億速云 閱讀:172 作者:Leah 欄目:編程語言

Java中怎么實現分布式計算,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

遠程過程調用的設計

要創建出4種東西:服務器、客戶端、服務器輔助設施和客戶端輔助設施.

1.創建客戶端和服務端應用程序,服務器應用程序時個遠程服務,是個帶有客戶端會調用的方法的對象

2.創建客戶端和服務器端的輔助設施(helper)他們會處理所有客戶端和服務器的底層網絡輸入/輸出細節,讓客戶端和程序好像在處理本地調用一樣.

輔助設施的任務輔助設施是個在實際上執行通信的對象,他們會讓客戶端感覺上好像是在調用本機對象,客戶端對象看起來像是在調用遠程的方法,但實際上它只是在調用本地處理Socket和串流細節的代理.在服務器這端,服務器的輔助設施會通過socket連接來自客戶端設施的要求,解析打包送來的信息,然后調用真正的服務,因此對服務對象來說此調用來自本地.服務的輔助設施取得返回值之后就把它包裝然后送回去(通過socket的輸出串流)給客戶端的輔助設施.客戶端的輔助設施會解開這些信息傳輸給客戶端的對象

調用方法的過程

1.客戶端對象對輔助設施對象調用doBigThing()

2.客戶端輔助設施把調用信息打包通過網絡送到服務器的輔助設施

3.服務端的輔助設施解開來自客戶端輔助設施的信息,并以此調用真正的服務.

這個過程的描述圖如下:

Java中怎么實現分布式計算

Java RMI提供客戶端和服務器端的輔助設施對象

在Java中,RMI已經幫我們創建好客戶端和服務器端的輔助設施,它也知道如何讓客戶端輔助設施看起來像是真正的服務,也就是說,RMI知道如何提供相同的方法給客戶端調用.

此外,RMI有提供執行期所需全部的基礎設施,包括服務的查詢以及讓客戶端能夠找到與取得客戶端的輔助設施(真正的服務代理人).

使用RMI時,無需編寫任何網絡或輸入/輸出的程序,客戶端對遠程方法的調用就跟對同一個Java虛擬機上的方法調用是一樣的.

一般調用和RMI調用有一點不同,雖然對客戶端來說,此方法調用看起來像是本地的,但是客戶端輔助設施會通過網絡發出調用,此調用最終還是會涉及到socket和串流,一開始是本機調用,代理會把它轉成遠程的.中間的信息是如何從Java虛擬機送到Java虛擬機要看輔助設施對象所用的協議而定.

使用RMI時,必須要決定協議:JRMP或IIOP,JRMP是RMI原生的協議,它是為Java間的遠程調用而設計的,另外一方面,IIOP是為了CORBA而產生的,它讓我們能夠調用Java對象或其它類型的遠程方法,CORBA通常比RMI麻煩,因為若兩端不全都是Java的話,就會產生一堆可怕的轉譯和交談操作.

我們只關心Java對Java的操作,所以會使用相當簡易的RMI.

在RMI中,客戶端的輔助設施稱為stub,而服務器端的輔助設施稱為skeleton.

如何創建遠程服務

1.創建Remote接口

遠程的接口定義了客戶端可以遠程調用的方法,它是個作為服務的多態化類.stub和服務都會實現此接口

2.實現Remote接口

這個是真正執行的類,它實現出定義在該接口上的方法,它是客戶端會調用的對象

3.用rmic產生stub和skeleton

客戶端和服務器都有helper,我們無需創建這些類或產生這些類的源代碼,這都會在執行JDK所附的rmic工具時自動地處理掉

4.啟動RMI registry (rmiregistry)

rmiregistry就像電話薄,用戶會從此處取得代理(客戶端的stub/helper對象)

5.啟動遠程服務

必須讓服務對象開始執行,實現服務的類會起始服務的實例并向RMI Registry注冊,要有注冊后才能對用戶服務.

服務端代碼

定義接口

import java.rmi.Remote;  import java.rmi.RemoteException;   /**   *    *    MyRemote.java   *   *     功   能: TODO    *     類   名: MyRemote.java   *   *  ver     変更日       角色    擔當者     変更內容   *     ──────────────────────────────────────────────   *  V1.00   2013-3-19   模塊    蘇若年     初版   *   *     Copyright (c) 2013 dennisit corporation All Rights Reserved.   *      *  Email:<a href="mailto:DennisIT@163.com">發送郵件</a>   *     *     *     Remote是個標記性的接口,意味著沒有方法,然而它對RMI有特殊的意義,所以必須遵守這項規則,   *     注意這里用的是extends,接口是可以繼承其他接口的   *    */ public interface MyRemote extends Remote{            /**       * 遠程的接口定義了客戶端可以遠程調用的方法,它是作為服務的多態化類,也就是說,客戶端會       * 調動有實現此接口的stub,而此stub因為會執行網絡和輸入/輸出工作,所以可能會發生各種       * 問題,客戶端鼻息處理或聲明異常來認知這一類風險,如果該方法在接口中聲明異常,調用該方       * 法的所有程序都必須處理或再聲明此異常.       *        * 遠程方法的參數和返回值必須是primitive或serializable的.任何遠程方法的參數都會被       * 打包通過網絡傳送,而這時通過序列化完成的,返回值也是一樣.所以,如果使用的是自定義類型       * 時,必須對其序列化       * @return       * @throws RemoteException           *                         所有接口中的方法都必須聲明RemoteException       */     public String sayHello() throws RemoteException;            }

業務實現

import java.rmi.Naming;  import java.rmi.RemoteException;  import java.rmi.server.UnicastRemoteObject;   /**   *    *    MyRemoteImpl.java   *   *     功   能: TODO    *     類   名: MyRemoteImpl.java   *   *  ver     変更日       角色    擔當者     変更內容   *     ──────────────────────────────────────────────   *  V1.00   2013-3-19   模塊    蘇若年     初版   *   *     Copyright (c) 2013 dennisit corporation All Rights Reserved.   *      *  Email:<a href="mailto:DennisIT@163.com">發送郵件</a>   *     *  為了要成為遠程服務對象,對象必須要有與遠程有關的功能,其中最簡單的方法就是繼承UnicastRemoteObject   *  (來自java.rmi.server)以讓這個父類處理這些工作   *   */ public class MyRemoteImpl extends UnicastRemoteObject implements MyRemote{       /**       * 父類的構造函數聲明了異常,所有你必須寫出構造函數,因為它代表你的構造函數會調用有風險的程序代碼       *        * UnicastRemoteObject有個小問題,它的構造函數會拋出RemoteException.處理它的***方式就是       * 對自己的實現聲明一個構造,如此才會有地方可以聲明出RemoteException.當類被初始化的時候,父類       * 的構造函數一定會被調用,如果父類的構造函數拋出異常,我們也必須聲明的自定義的構造函數會拋出異常       * @throws RemoteException       */     protected MyRemoteImpl() throws RemoteException {       }       /**       * 實現出接口所有的方法,但無需聲明RemoteException       */     @Override     public String sayHello(){          return "server says, rmi hello world !";      }       public static void main(String[] args) {          try {              /**               * 我們已經有了遠程服務,還必須要讓遠程用戶存取,這可以通過將它初始化并加進RMI Registry               * (它一定要運行起來,不然此程序就會失敗).當注冊對象時,RMI系統會把stub加到registry中,               * 因為這是客戶端所需要的.使用java.rmi.Naming的rebind()來注冊服務               */             MyRemote service = new MyRemoteImpl();              /**               * 創建出遠程對象,然后使用靜態的Naming.rebind()來產生關聯,所注冊的名稱會提供客戶端查詢               */             Naming.rebind("Remote Hello World", service);          } catch (Exception e) {              e.printStackTrace();          }      }        }

客戶端代碼

import java.rmi.Naming;  /**   *    *    MyRemoteClient.java   *   *     功   能: TODO    *     類   名: MyRemoteClient.java   *   *  ver     変更日       角色    擔當者     変更內容   *     ──────────────────────────────────────────────   *  V1.00   2013-3-19   模塊    蘇若年     初版   *   *     Copyright (c) 2013 dennisit corporation All Rights Reserved.   *      *  Email:<a href="mailto:DennisIT@163.com">發送郵件</a>   *   */ public class MyRemoteClient {       public void exec(){          try {              /**               * 客戶端必須取得stub對象,因為客戶端必須要調用它的方法.這就得靠RMI registry了.客戶端會像查詢電話               * 簿一樣地搜索,找出上面有相符的名稱的服務.               * 客戶端查詢RMIRegistry,返回stub對象               * Naming.lookup("rmi://127.0.0.1/Remote Hello World");               * 參數說明               * rmi://127.0.0.1/Remote Hello World               * 127.0.0.1表示主機名稱或主機IP地址               * Remote Hello World必須要跟注冊的名稱一樣               *                */             MyRemote service = (MyRemote)Naming.lookup("rmi://127.0.0.1/Remote Hello World");              String tmp = service.sayHello();              System.out.println(tmp);          } catch (Exception e) {              e.printStackTrace();          }      }            public static void main(String[] args) {          new MyRemoteClient().exec();      }  }

對實現出的類(不是remote接口)執行rmic

伴隨JDK而來的rmic工具會以服務的實現產生2個心的類stub和skeleton.它會按照命名規則在遠程實現名稱后面加上_Stub或_Skeleton。rmic有幾個選項,包括了不產生skeleton、觀察產生出類的源代碼或使用IIOP作為通訊協議等.產生出的類會放在當前目錄下,要記住rmic必須能夠找到所實現的類,因此可能要從實現所在的目錄執行rmic(實際中可能需要考慮到包目錄結構和完整名稱,為了簡便這里沒有運用到包)

調用命令行來啟動rmiregistry,要確定是從可以存取到該類的目錄來啟動,最簡單的方法就是從類這個目錄來運行.

運行截圖如下

Java中怎么實現分布式計算

注意:

客戶端是使用接口來調用stub上的方法,客戶端的Java虛擬機必須要有stub類,但客戶端不會在程序代碼中引用到stub類,客戶端總是通過接口來操作真正的遠程對象

服務器上必須要有stub和skeleton,以及服務與遠程的接口,它會需要stub類是因為stub會被代換成連接在RMIRegistry上真正的服務.

使用RMI時常犯的錯誤:

1.忘記在啟動遠程服務錢啟動rmiregistry(使用Naming.rebind()注冊服務前rmiregistry必須啟動)

2.忘記把參數和返回類型做成可序列化(編譯不會檢測到,執行時才會發現)

3.忘記將stub類交給客戶端

RMI很適合編寫并運行遠程服務,但我們不會單獨使用RMI來執行網站服務,對大型的企業級應用程序來說,我們需要更多更好的功能.像交易管理、大量并發處理、安全性和數據庫管理等.這就需要用到Enterprise Application Server.

JavaEE服務器包括了Web服務器和Enterprise JavaBeans(EJB)服務器. EJB服務器作用于RMI調用和服務層之間.

RMI在JINI中的應用

Jini也是使用RMI(雖然也可以用別的協議),但多了幾個關鍵功能.

1.自適應探索(adaptive discovery)

2.自恢復網絡(self-healing networks)

RMI的客戶端得先取得遠程服務的地址和名稱.客戶端的查詢程序代碼就要帶有遠程服務的IP地址或主機名(因為RMIRegistry就在上面)以及服務所注冊的名稱

但是用JINI時,用戶只需要知道一件事,服務所實現的接口!這樣就行.

Jini是用lookup service,該查詢服務比RMI Registry更強更有適應性.因為Jini會在網絡上自動的廣告.當查詢服務上線是,它會使用IP組播技術送出信息給整個網絡.不止這樣,如果客戶端在查詢服務已經廣播之后上線,客戶端也可以發出消息給整個網絡來詢問.

當服務上線時,它會動態的探索網絡上的JINI查詢服務并申請注冊,注冊時,服務會送出一個序列化的對象給查詢服務,此對象可以是RMI遠程服務的stub、網絡裝置的驅動程序,甚或是可以在客戶端執行的服務本身.并且注冊的是所實現的接口.而不是名稱.

自適應探索的運作

1.Jini查詢服務在網絡上啟動,并使用IP組播技術為自己做宣傳

2.已經啟動的另外一個Jini服務會尋求向剛啟動的查詢服務注冊.它注冊的是功能而不是名稱,也就是所實現的接口,然后送出序列化對象給查詢服務

3.網絡客戶想要取得實現ScientificCalculator的東西,可是不知道哪里有,所以就問查詢服務

4.查詢服務響應查詢的結果

自恢復網絡的運作

1.某個Jini服務要求注冊,查詢服務會給一份租約,新注冊的服務必須要定期更新租約,不然查詢服務會假設此服務已經離線了,查詢服務會力求呈現精確完整的可用服務網絡狀態

2.因為關機所以服務離線,因此沒有更新租約,查詢服務就把它踢掉.

看完上述內容是否對您有幫助呢?如果還想對相關知識有進一步的了解或閱讀更多相關文章,請關注億速云行業資訊頻道,感謝您對億速云的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

府谷县| 衡阳市| 蓝田县| 田阳县| 镇江市| 五台县| 黄浦区| 辽中县| 灵丘县| 太康县| 湖州市| 隆昌县| 太仆寺旗| 玉门市| 木兰县| 观塘区| 扬中市| 合肥市| 尖扎县| 巴南区| 屏东市| 博野县| 湾仔区| 吉木萨尔县| 河北省| 介休市| 凤翔县| 昭平县| 牡丹江市| 凤城市| 绵阳市| 漳州市| 吉林省| 波密县| 梨树县| 贡山| 平邑县| 翁牛特旗| 天全县| 涿州市| 高密市|