亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術(shù)文章
文章詳情頁

java RMI詳細(xì)介紹及實(shí)例講解

瀏覽:4日期:2022-08-28 14:16:29

java本身提供了一種RPC框架——RMI(即RemoteMethodInvoke遠(yuǎn)程方法調(diào)用),在編寫一個接口需要作為遠(yuǎn)程調(diào)用時,都需要繼承了Remote,Remote接口用于標(biāo)識其方法可以從非本地虛擬機(jī)上調(diào)用的接口,只有在“遠(yuǎn)程接口”(擴(kuò)展java.rmi.Remote的接口)中指定的這些方法才可遠(yuǎn)程使用,下面通過一個簡單的示例,來講解RMI原理以及開發(fā)流程:

為了真正實(shí)現(xiàn)遠(yuǎn)程調(diào)用,首先創(chuàng)建服務(wù)端工程rmi-server,結(jié)構(gòu)如下:

java RMI詳細(xì)介紹及實(shí)例講解 

代碼說明:

1.User.java:用于遠(yuǎn)程調(diào)用時pojo對象的傳輸,該對象必須實(shí)現(xiàn)Serializable接口,否則在調(diào)用過程中,會拋出NotSerializableException異常,代碼如下:

/** * 用戶信息,用于遠(yuǎn)程調(diào)用傳輸,必須實(shí)現(xiàn)Serializable接口 * * @author andy * */public class User implements Serializable { private static final long serialVersionUID = 1L; private String name; private int age; public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } @Override public String toString() { return 'name : ' + this.name + ', age : ' + this.age; }}

2.Hello.java:遠(yuǎn)程接口,該接口需要繼承Remote接口,并且接口中的方法全都要拋出RemoteException異常,代碼如下:

import java.rmi.Remote;import java.rmi.RemoteException;import pers.andy.rmi.bean.User;/** * 定義一個遠(yuǎn)程接口,必須繼承Remote接口,其中需要遠(yuǎn)程調(diào)用的方法必須拋出RemoteException異常 * * @author andy * */public interface IHello extends Remote { /** * 更新user信息 * @param user * @return * @throws RemoteException */ public User updateUser(User user) throws RemoteException;}

3.HelloImpl:遠(yuǎn)程接口實(shí)現(xiàn)類,必須繼承UnicastRemoteObject(繼承RemoteServer->繼承RemoteObject->實(shí)現(xiàn)Remote,Serializable),只有繼承UnicastRemoteObject類,才表明其可以作為遠(yuǎn)程對象,被注冊到注冊表中供客戶端遠(yuǎn)程調(diào)用(補(bǔ)充:客戶端lookup找到的對象,只是該遠(yuǎn)程對象的Stub(存根對象),而服務(wù)端的對象有一個對應(yīng)的骨架Skeleton(用于接收客戶端stub的請求,以及調(diào)用真實(shí)的對象)對應(yīng),Stub是遠(yuǎn)程對象的客戶端代理,Skeleton是遠(yuǎn)程對象的服務(wù)端代理,他們之間協(xié)作完成客戶端與服務(wù)器之間的方法調(diào)用時的通信。),代碼如下:

/** * 遠(yuǎn)程的接口的實(shí)現(xiàn),繼承了UnicastRemoteObject,表明該類作為一個遠(yuǎn)程對象 * * @author andy * */public class HelloImpl extends UnicastRemoteObject implements IHello { /** * */ private static final long serialVersionUID = 1L; /** * 因?yàn)閁nicastRemoteObject的構(gòu)造方法拋出了RemoteException異常,因此這里默認(rèn)的構(gòu)造方法必須寫,必須聲明拋出RemoteException異常 * * @throws RemoteException */ public HelloImpl() throws RemoteException { } public User updateUser(User user) throws RemoteException { System.out.println('-------------- 客戶端發(fā)送的user為' + user.toString()); user.setName('andy2'); user.setAge(30); return user; }}

4.HelloServer:服務(wù)端啟動類,用于創(chuàng)建遠(yuǎn)程對象注冊表以及注冊遠(yuǎn)程對象,代碼如下:

/** * 服務(wù)端啟動類 * * @author andy * */public class HelloServer { public static void main(String args[]) { try { // 本地主機(jī)上的遠(yuǎn)程對象注冊表Registry的實(shí)例,并指定端口為8888,這一步必不可少(Java默認(rèn)端口是1099) LocateRegistry.createRegistry(8888); // 把遠(yuǎn)程對象注冊到RMI注冊服務(wù)器上,并命名為RHello // 綁定的URL標(biāo)準(zhǔn)格式為:rmi://host:port/name(其中協(xié)議名可以省略,下面兩種寫法都是正確的) Naming.bind('rmi://localhost:8888/RHello', rhello); // Naming.bind('//localhost:8888/RHello',rhello); System.out.println('------------遠(yuǎn)程對象IHello注冊成功,等待客戶端調(diào)用...'); } catch (RemoteException e) { System.out.println('創(chuàng)建遠(yuǎn)程對象發(fā)生異常!'); e.printStackTrace(); } catch (AlreadyBoundException e) { System.out.println('發(fā)生重復(fù)綁定對象異常!'); e.printStackTrace(); } catch (MalformedURLException e) { System.out.println('發(fā)生URL畸形異常!'); e.printStackTrace(); } }}

補(bǔ)充說明:為何HelloImpl繼承了UnicastRemoteObject就可以被作為遠(yuǎn)程對象發(fā)布,查閱UnicastRemoteObject的源碼可以發(fā)現(xiàn):

protected UnicastRemoteObject() throws RemoteException { this(0); } protected UnicastRemoteObject(int port) throws RemoteException { this.port = port; exportObject((Remote) this, port); }

其實(shí)在啟動server端的時候,new了HelloImpl對象,因?yàn)槔^承了UnicastRemoteObject,會先調(diào)用父類的構(gòu)造方法,這時候,就會將this(當(dāng)前對象)通過exportObject方法注冊。

所以,如果在被導(dǎo)出的對象需要繼承其它的類,那么就可以不采用集成UnicastRemoteObject的方式,而是通過exportObject方法將其導(dǎo)出為遠(yuǎn)程對象:

...// 創(chuàng)建一個遠(yuǎn)程對象IHello rhello = new HelloImpl();//HelloImpl不需要繼承UnicastRemoteObject類,通過exportObject將其顯示導(dǎo)出UnicastRemoteObject.exportObject(rhello,0);...

以上即是服務(wù)端所有代碼,接下來是創(chuàng)建客戶端工程,結(jié)構(gòu)如下:

java RMI詳細(xì)介紹及實(shí)例講解

實(shí)際應(yīng)用開發(fā)中,客戶端的User.java和IHello.java應(yīng)該是從服務(wù)端導(dǎo)出jar包的形式添加到依賴庫里,因此這邊只介紹HelloClient.java,該類為客戶端啟動類,用于在注冊表中查找遠(yuǎn)程對象實(shí)現(xiàn)遠(yuǎn)程方法調(diào)用,代碼如下:

/** * 客戶端啟動類 * * @author andy * */public class HelloClient { public static void main(String args[]) { try { // 在RMI服務(wù)注冊表中查找名稱為RHello的對象,并調(diào)用其上的方法 IHello rhello = (IHello) Naming.lookup('rmi://localhost:8888/RHello'); // 構(gòu)造user對象,測試遠(yuǎn)程對象傳輸 User user = new User(); user.setAge(20); user.setName('andy'); System.out.println('-------------- 服務(wù)端返回的的user為' + rhello.updateUser(user).toString()); } catch (NotBoundException e) { e.printStackTrace(); } catch (MalformedURLException e) { e.printStackTrace(); } catch (RemoteException e) { e.printStackTrace(); } }}

到此為止,客戶端和服務(wù)端的工程都搭建完畢,現(xiàn)在可以進(jìn)行測試,執(zhí)行次序和測試結(jié)果如下所示:

1.首先運(yùn)行服務(wù)端啟動類HelloServer,結(jié)果如下:

服務(wù)端:------------遠(yuǎn)程對象IHello注冊成功,等待客戶端調(diào)用...

2.運(yùn)行客戶端啟動類,結(jié)果如下:

服務(wù)端:-------------- 客戶端發(fā)送的user為name : andy, age : 20 客戶端:-------------- 服務(wù)端返回的的user為name : andy2, age : 30

 到此這篇關(guān)于java RMI詳細(xì)介紹及實(shí)例講解的文章就介紹到這了,更多相關(guān)java RMI 內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)! 

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 美女国产网站 | 国产精品对白刺激久久久 | 久久亚洲精品一区成人 | 免费看成人国产一区二区三区 | 国产视频播放 | 免费人成网站在线播放 | 综合亚洲一区二区三区 | 亚洲一区二区三区高清视频 | 亚洲色图在线视频 | 黑人猛男大战俄罗斯白妞 | 国产福利免费在线观看 | 国产xxxxxx久色视频在 | 中文字幕日韩在线 | 亚洲第一区第二区 | 涩色网 | 国内精品自在自线视频香蕉 | 国产亚洲精品久久久久久无 | 亚洲欧洲色图 | 亚洲国产精品综合久久 | 日本wwwwwxxxxx | 国产一区二区三区四区20p | 色狠狠婷婷97 | 久久99精品视频在线在线观看 | 成人在线视频国产 | 久久精品国产亚洲香蕉 | 国产91短视频 | 最近中文字幕免费完整 | 91精品在线看 | 免费的黄色的视频 | 不卡一区 | 欧美黄色一级大片 | 亚洲综合免费 | 天天影视色 | 成人性色生活片免费看爆迷你毛片 | 一级做α爱过程免费视频 | 一级二级三级黄色片 | 亚洲精品区在线播放一区二区 | 五月激激激综合网色播免费 | 一级毛片在线播放 | 日本黄色一区 | 国产成人夜间影院在线观看 |