欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2020年07月23日 10:05:03   作者:yehx  
這篇文章主要介紹了java RMI詳細(xì)介紹及實(shí)例講解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

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

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

 

  代碼說明:

  1.User.java:用于遠(yuǎn)程調(diào)用時(shí)pojo對(duì)象的傳輸,該對(duì)象必須實(shí)現(xiàn)Serializable接口,否則在調(diào)用過程中,會(huì)拋出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;

/**
 * 定義一個(gè)遠(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)程對(duì)象,被注冊(cè)到注冊(cè)表中供客戶端遠(yuǎn)程調(diào)用(補(bǔ)充:客戶端lookup找到的對(duì)象,只是該遠(yuǎn)程對(duì)象的Stub(存根對(duì)象),而服務(wù)端的對(duì)象有一個(gè)對(duì)應(yīng)的骨架Skeleton(用于接收客戶端stub的請(qǐng)求,以及調(diào)用真實(shí)的對(duì)象)對(duì)應(yīng),Stub是遠(yuǎn)程對(duì)象的客戶端代理,Skeleton是遠(yuǎn)程對(duì)象的服務(wù)端代理,他們之間協(xié)作完成客戶端與服務(wù)器之間的方法調(diào)用時(shí)的通信。),代碼如下:

/**
 * 遠(yuǎn)程的接口的實(shí)現(xiàn),繼承了UnicastRemoteObject,表明該類作為一個(gè)遠(yuǎn)程對(duì)象
 * 
 * @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ù)端啟動(dòng)類,用于創(chuàng)建遠(yuǎn)程對(duì)象注冊(cè)表以及注冊(cè)遠(yuǎn)程對(duì)象,代碼如下:

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

  補(bǔ)充說明:為何HelloImpl繼承了UnicastRemoteObject就可以被作為遠(yuǎn)程對(duì)象發(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í)在啟動(dòng)server端的時(shí)候,new了HelloImpl對(duì)象,因?yàn)槔^承了UnicastRemoteObject,會(huì)先調(diào)用父類的構(gòu)造方法,這時(shí)候,就會(huì)將this(當(dāng)前對(duì)象)通過exportObject方法注冊(cè)。

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

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

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

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

/**
 * 客戶端啟動(dòng)類
 * 
 * @author andy
 *
 */
public class HelloClient {
  public static void main(String args[]) {
    try {
      // 在RMI服務(wù)注冊(cè)表中查找名稱為RHello的對(duì)象,并調(diào)用其上的方法
      IHello rhello = (IHello) Naming.lookup("rmi://localhost:8888/RHello");       // 構(gòu)造user對(duì)象,測(cè)試遠(yuǎn)程對(duì)象傳輸
      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)行測(cè)試,執(zhí)行次序和測(cè)試結(jié)果如下所示:

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

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

  2.運(yùn)行客戶端啟動(dòng)類,結(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)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家! 

相關(guān)文章

  • 基于SpringIOC創(chuàng)建對(duì)象的四種方式總結(jié)

    基于SpringIOC創(chuàng)建對(duì)象的四種方式總結(jié)

    這篇文章主要介紹了基于SpringIOC創(chuàng)建對(duì)象的四種方式總結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • Mybatis動(dòng)態(tài)Sql標(biāo)簽使用小結(jié)

    Mybatis動(dòng)態(tài)Sql標(biāo)簽使用小結(jié)

    本文主要介紹了Mybatis動(dòng)態(tài)Sql標(biāo)簽使用,常用的動(dòng)態(tài)sql標(biāo)簽包括?if、choose(when、otherwise)、trim(where、set)、foreach,下面就來介紹一下
    2024-04-04
  • java使用MulticastSocket實(shí)現(xiàn)多點(diǎn)廣播

    java使用MulticastSocket實(shí)現(xiàn)多點(diǎn)廣播

    這篇文章主要為大家詳細(xì)介紹了java使用MulticastSocket實(shí)現(xiàn)多點(diǎn)廣播,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-01-01
  • SpringBoot創(chuàng)建RSocket服務(wù)器的全過程記錄

    SpringBoot創(chuàng)建RSocket服務(wù)器的全過程記錄

    RSocket應(yīng)用層協(xié)議支持 Reactive Streams語義, 例如:用RSocket作為HTTP的一種替代方案。這篇文章主要給大家介紹了關(guān)于SpringBoot創(chuàng)建RSocket服務(wù)器的相關(guān)資料,需要的朋友可以參考下
    2021-05-05
  • springboot動(dòng)態(tài)定時(shí)任務(wù)的實(shí)現(xiàn)方法示例

    springboot動(dòng)態(tài)定時(shí)任務(wù)的實(shí)現(xiàn)方法示例

    這篇文章主要給大家介紹了關(guān)于springboot動(dòng)態(tài)定時(shí)任務(wù)的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • java設(shè)計(jì)模式之建造者模式學(xué)習(xí)

    java設(shè)計(jì)模式之建造者模式學(xué)習(xí)

    建造者模式(Builder Pattern)主要用于“分步驟構(gòu)建一個(gè)復(fù)雜的對(duì)象”,在這其中“分步驟”是一個(gè)穩(wěn)定的算法,下面給出了詳細(xì)的示例
    2014-01-01
  • 關(guān)于SpringBoot啟動(dòng)速度慢的原因總結(jié)

    關(guān)于SpringBoot啟動(dòng)速度慢的原因總結(jié)

    這篇文章主要介紹了關(guān)于SpringBoot啟動(dòng)速度慢的原因總結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • Spring配置動(dòng)態(tài)數(shù)據(jù)源實(shí)現(xiàn)讀寫分離的方法

    Spring配置動(dòng)態(tài)數(shù)據(jù)源實(shí)現(xiàn)讀寫分離的方法

    這篇文章主要介紹了利用Spring配置動(dòng)態(tài)數(shù)據(jù)源實(shí)現(xiàn)讀寫分離的方法,文中通過示例代碼介紹的很詳細(xì),相信對(duì)大家的理解和學(xué)習(xí)具有一定的參考借鑒價(jià)值,藕需要的朋友可以一起學(xué)習(xí)學(xué)習(xí)。
    2017-01-01
  • JAVA+Hibernate 無限級(jí)分類

    JAVA+Hibernate 無限級(jí)分類

    主要看menu_id和parent_id這兩個(gè)字段。 Eclipse生成的表持久映射:(說明:自己加level屬性,作用:為了記錄種類所在深度)
    2008-07-07
  • springBoot系列常用注解(小結(jié))

    springBoot系列常用注解(小結(jié))

    這篇文章主要介紹了springBoot系列常用注解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04

最新評(píng)論