Java 網路編程 —— RMI 框架

来源:https://www.cnblogs.com/Yee-Q/archive/2023/06/15/17483415.html
-Advertisement-
Play Games

## 概述 RMI 是 Java 提供的一個完善的簡單易用的遠程方法調用框架,採用客戶/伺服器通信方式,在伺服器上部署了提供各種服務的遠程對象,客戶端請求訪問伺服器上遠程對象的方法,它要求客戶端與伺服器端都是 Java 程式 RMI 框架採用代理來負責客戶與遠程對象之間通過 Socket 進行通信的 ...


概述

RMI 是 Java 提供的一個完善的簡單易用的遠程方法調用框架,採用客戶/伺服器通信方式,在伺服器上部署了提供各種服務的遠程對象,客戶端請求訪問伺服器上遠程對象的方法,它要求客戶端與伺服器端都是 Java 程式

RMI 框架採用代理來負責客戶與遠程對象之間通過 Socket 進行通信的細節。RMI 框架為遠程對象分別生成了客戶端代理和伺服器端代理。位於客戶端的代理必被稱為存根(Stub),位於伺服器端的代理類被稱為骨架(Skeleton)

當客戶端調用遠程對象的一個方法時,實際上是調用本地存根對象的相應方法。存根對象與遠程對象具有同樣的介面。存根採用一種與平臺無關的編碼方式,把方法的參數編碼為位元組序列,這個編碼過程被稱為參數編組。RMI 主要採用Java 序列化機制進行參數編組。存根把以下請求信息發送給伺服器:

  • 被訪問的遠程對象的名字
  • 被調用的方法的描述
  • 編組後的參數的位元組序列

伺服器端接收到客戶端的請求信息,然後由相應的骨架對象來處理這一請求信息,骨架對象執行以下操作:

  • 反編組參數,即把參數的位元組序列反編碼為參數
  • 定位要訪問的遠程對象
  • 調用遠程對象的相應方法
  • 獲取方法調用產生的返回值或者異常,然後對它進行編組
  • 把編組後的返回值或者異常發送給客戶

客戶端的存根接收到伺服器發送過來的編組後的返回值或者異常,再對它進行反編組,就得到調用遠程方法的返回結果

JDK5.0 之後,RMI 框架會在運行時自動為運程對象生成動態代理類(包括存根和骨架類),從而更徹底地封裝了 RMI 框架的實現細節,簡化了 RMI 框架的使用方式


創建 RMI 應用

創建一個 RMI 應用包括以下步驟:

  • 創建遠程介面:繼承 java.rmi.Remote 介面
  • 創建遠程類:實現遠程介面
  • 創建伺服器程式:負責在 RMI 註冊器中註冊遠程對象
  • 創建客戶程式:負貴定位遠程對象,並且調用遠程對象的方法

1. 創建遠程介面

遠程介面中聲明瞭可以被客戶程式訪問的遠程方法,並直接或間接繼承 java.rmi.Remote 介面

import java.rmi.*;

public interface HelloService extends Remote {
    public String echo(String msg) throws RemoteException;
}

2. 創建遠程類

遠程類必須實現一個遠程介面,此外,為了使遠程類的實例變成能為遠程客戶提供服務的遠程對象,可通過以下兩種途徑之一把它導出為遠程對象:

  • 使遠程類繼承 java.rmi.server.UnicastRemoteObjcct 類,並且遠程類的構構方法必聲明拋出 RemoteException

    import java.rmi.*;
    import java.rmi.server.UnicastRemoteObjoct;
    
    public class HelloServlceImpl extends UnicagtRemoteObject implements HelloService {
        
        private String name;
        
        public HelloServicelmpl(String name) throws RemoteException {
            this.name = name;
        }
        
        public String echo(String msg) throws RemoteException {
            System.out.println(name + ":測用echo()方法");
            return "echo;" + msg + " from" + name;
        }
    }
    
  • 如果一個遠程類已經繼承了其他類,無法再繼承 UnicastRemoteObiect 類,那麼可以在構造方法中調用 UnicastRemoteObject 類的靜態 expotObject 方法,同樣,遠程類的構造方法也必須聲明拋出 RemoteException

    public class HelloServlceImpl extends OtherClass implements HelloService {
        
        private String name;
        
        public HelloServicelmpl(String name) throws RemoteException {
            this.name = name;
            //參數 port 指定監聽的埠,如果取值為0,就表示監聽任意一個匿名埠
            UnicagtRemoteObject.exportobject(this, 0);
        }
        
        public String echo(String msg) throws RemoteException {
            System.out.println(name + ":測用echo()方法");
            return "echo;" + msg + " from" + name;
        }
    }
    

3. 創建伺服器程式

RMI 採用一種命名服務機制來使得客戶程式可以找到伺服器上的一個遠程對象,RMI註冊器提供這種命名服務。好比電話查詢系統,那些希望對外公開聯繫方式的單位先到查詢系統登記,當客戶想知道某個單位的聯繫方式時,只需向查詢系統提供單位的名字,查詢系統就會返回該單位的聯繫方式

啟動 RMI 註冊器有兩種方式。一種方式是直接運行 rmiregistry.exe 程式,在 JDK 的安裝目錄的 bin 子目錄下有一個 rmiregistry.exe 程式,它是提供命名服務的註冊器程式。儘管 rmiregistry 註冊器程式也可以單獨運行在一個主機上,但出於安全的原因,通常讓 rmiregistry 註冊器程式與伺服器程式運行在同一個主機上

啟動 RMI 註冊器的另一種方式是在伺服器程式中調用 java.rmiregistry.LocateRegistry 類的靜態方法 createRegistry()

//預設的監聽路口為1099
Registry registry = LocateRegistry.createRegigtry(1099);

向註冊器註冊遠程對象有三種方式:

//創建遠程對象
HelloService service1 = new HelloServiceImpl("service1");

//方式1:調用 java.i.registry.Registy 介面的 bind 或 rebind 方法
Registry registry = LocateRegistry.createRegistry(1099);
registry.rebind("HelloService1", service1);

//方式2:調用命名服務類 java.rmi.Naming 的 bind 或 rebind 方法
Naming.rebind("HelloService1", service1);

//方式3:調用 JNDI API 的 javax.naming.Context 介面的 bind 或rebind 方法
Context namingContext = new InitialContext();
namingContext.rebind("rmi:HelloService1", service1);

下例的 SimpleServer 類創建了兩個 HelloServicelmpl 遠程對象,接著創建並啟動 RMI 註冊器,然後把兩個遠程對象註冊到 RMI 註冊器

import java.rmi.*;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;

public class SimpleServer {
    
    public static void main( String args[]) {
        try {
            HelloService service1 = new HelloServiceImpl("service1");
            HelloService service2 = new HelloServiceImpl("service2");
            
            //創建並啟動註冊器
            Registry registry = LocateRegistry.createRegistry(1099);
            //註冊遠程對象
            regigtry.rebind("HelloService1", service1);
            regigtry.rebind("HelloService2", service2);
        } catch(Exception e) {
            e.printStackTrace();
        }
    }
}

關於向 RMI 註冊器註冊遠程對象,需要註意的是,遠程對象即使沒有在註冊器中註冊,也可被遠程訪問

4. 創建客戶程式

下例的 SimpleClient 類先獲得遠程對象的存根對象,接著調用它的遠程方法

import java.rmi.*;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;

public class SimpleClient {
    
    public static void main(String args[]) {
        try {
            //返回本地主機的RMI註冊器對象,參數port指定RMI註冊器監聽的埠
            Registry registry = LocateRegistry.getRegistry(1099);
            //查找對象,返回與參數name指定的名字所綁定的對象
            //返回的是一個名為"com.sun.proxy.$Proxy0"的動態代理類的實例
            HelloService service1 = (HelloService) registry.lookup("HelloService1");
            HelloService service2 = (HelloService) registry.lookup("HelloService2");
            
            System.out.println(service1.echo("hello"));
            System.out.println(service2.echo("hello"));
        }
    }
}

遠程方法中的參數與返回值傳遞

當客戶端調用伺服器端的遠程對象的方法時,客戶端會向伺服器端傳遞參數,伺服器端則會向客戶端傳遞返回值。RMI 規範對參數以及返回值的傳遞的規定如下所述:

  • 只有基本類型的數據、遠程對象以及可序列化的對象才可以被作為參數或者返回值進行傳遞
  • 如果參數或返回值是一個遠程對象,那麼把它的存根對象傳遞到接收方。也就是說接收方得到的是遠程對象的存根對象
  • 如果參數或返回值是可序列化對象,那麼直接傳遞該對象的序列化數據。也就是說接收方得到的是發送方的可序列化對象的複製品
  • 如果參數或返回值是基本類型的數據,那麼直接傳遞該數據的序列化數據。也就是說,接收方得到的是發送方的基本類型的數據的複製品

分散式垃圾收集

在 Java 虛擬機中,對於一個本地對象,只要不被本地 Java 虛擬機內的任何變數引用,它就會結束生命周期,可以被垃圾回收器回收。而對於一個遠程對象,不僅會被本地 Java 虛擬機內的變數引用,還會被遠程引用

伺服器端的一個遠程對象受到三種引用:

  • 伺服器端的一個本地對象持有它的本地引用
  • 這個遠程對象已經被註冊到 RMI 註冊器,可以理解為,RMI 註冊器持有它的引用
  • 客戶端獲得了這個遠程對象的存根對象,可以理解為,客戶端持有它的遠程引用

RMI 框架採用分散式垃圾收集機制來管理遠程對象的生命周期,當一個遠程對象不受到任何本地引用和遠程引用時,這個遠程對象才會結束生命周期,並且可以被本地 Java 虛擬機的垃圾回收器回收。

伺服器端如何知道客戶端持有一個遠程對象的遠程引用呢?當客戶端獲得了一個伺服器端的遠程對象的存根後,就會向伺服器發送一條租約通知,告訴伺服器自己持有這個遠程對象的引用了。客戶端對這個遠程對象有一個租約期限,預設值為 600000ms。當至達了租約期限的一半時間,客戶如果還持有遠程引用,就會再次向伺服器發送租約通知。客戶端不斷在給定的時間間隔中向伺服器發送租約通知,從而使腸務器知道客戶端一直持有遠程對象的引用。如果在租約到期後,伺服器端沒有繼續收到客戶端的新的租約通知,伺服器端就會認為這個客戶已經不再持有遠程對象的引用了


動態載入

遠程對象一般分佈在伺服器端,當客戶端試圖調用遠程對象的方法時,如果在客戶端還不存在遠程對象所依賴的類文件,比如遠程方法的參數和返回值對應的類文件,客戶就會從 java.rmi.server.codebase 系統屬性指定的位貿動態載入該類文件

同樣,當伺服器端訪問客戶端的遠程對象時,如果伺服器端不存在相關的類文件,腐務器就會從 java.rmi.server.codebase 屬性指定的位置動態載入它們

此外,當伺服器向 RMI 註冊器註冊遠程對象時,註冊器也會從 java.rmi.server.codebase 屬性指定的位置動態載入相關的遠程介面的類文件

前面的例子都是在同一個 classpath 下運行伺服器程式以及客戶程式的,這些程式都能從本地 classpath 中找到相應的類文件,因此無須從 java.rmi.server.codebase 屬性指定的位置動態載入類。而在實際應用中,客戶程式與伺服器程式運行在不同的主機上,因此當客戶端調用伺服器端的遠程對象的方法時,有可能需要從遠程文件系統載入類文件。同樣,當伺服器端調用客戶端的遠程對象的方法時,也有可能從遠程文件系統載入類文件

我們可以且把這些需要被載入的類的文件都集中放在網路上的同一地方,啟動時將java.rmi.server.codebase 設置為指定位置,從而實現動態載入

start java -Djava.rmi.server.codebase=http://www.javathinker.net/download/


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 函數並非總是直接顯示輸出,它可以處理一些數據,並返回一個或一組值。函數返回的值被稱為返回值。 在函數中,可使用return關鍵字將值返回到調用函數的代碼行。返回值能夠將程式的大部分繁重工作移到函數中去,從而簡化主程式。函數可返回任何類型的值,包括列表和字典等較複雜的數據結構。 # 1.返回簡單值 返 ...
  • 某日二師兄參加XXX科技公司的C++工程師開發崗位第15面: > 面試官:瞭解`auto`關鍵字嗎? > > 二師兄:嗯,瞭解一些(我很熟悉)。 > > 面試官:說一說`auto`的用法吧? > > 二師兄:`auto`主要是為了編譯器進行類型推導。比如: ```c++ auto i = 42; / ...
  • 對於一個直接創建的springBoot項目工程來說,可以按照以下步驟使用rabbitmq - 添加依賴:添加rabbitMQ的依賴。 ~~~java org.springframework.boot spring-boot-starter-amqp ~~~ - 配置連接:在配置文件中配置虛擬主機、端 ...
  • 鑒於函數定義中可能包含多個形參變數,因此函數調用中也可能包含多個實參變數。向函數傳遞實參變數給形參變數的方式有很多,可使用**位置參數**,這要求實參變數的順序與形參變數的順序相同;也可使用**關鍵字參數**,都由變數名和值組成,簡稱名稱-值對;還可使用**列表(元組)和字典**。 # 1.位置參數 ...
  • # scala的基本語法 ## 註釋 對於scala的註釋,簡而言之就是一句話,和java的註釋一模一樣 基本語法 ``` (1)單行註釋:// (2)多行註釋:/* */ (3)文檔註釋:/** * */ ``` 代碼示例: ```Scala package com.doitedu.demo01 ...
  • > 本文首發於公眾號:Hunter後端 > 原文鏈接:[celery筆記四之在Django中使用celery](https://mp.weixin.qq.com/s/O-vHxvDbU6aSGM63GwLLdw) 這一篇筆記介紹一下如何在 Django 系統中使用 celery。 如果是想純粹使用 ...
  • # 網格線分級 一般情況下,我們設置網格線都不會太在意密度和精度的問題,導致下圖這種看的有些眼花,橫坐標在比較密集的時候很容易看竄行。 ![image-20230615191231845](https://img2023.cnblogs.com/blog/2862884/202306/2862884 ...
  • 分析 Switch 相較於 if 的優點 1、switch 執行效率 高於 if 的執行效率 分析: switch是在編譯階段將子函數的地址和判斷條件綁定了,只要直接將a的直接映射到子函數地址去執行就可以了, if處理起來首先要把a的值放到CPU的寄存器中,然後要把比較的值放到CPU的另一個寄存器中 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...