Java RMI(远程方法调用)开发

简介: 参考 https://docs.oracle.com/javase/7/docs/platform/rmi/spec/rmi-arch2.html http://www.cnblogs.com/wxisme/p/5296441.

参考

https://docs.oracle.com/javase/7/docs/platform/rmi/spec/rmi-arch2.html

http://www.cnblogs.com/wxisme/p/5296441.html

http://blog.csdn.net/qb2049_xg/article/details/3278672 

http://classfoo.com/ccby/article/p1wgbVn

http://www.cnblogs.com/yin-jingyu/archive/2012/06/14/2549361.html

RMI与RPC

RMI在我看来更像Java专属的RPC,或者说纯面向对象的RPC。跟RPC一样是分布式非常重要的内容,也是Java消息中间件的基础。

RMI原理

本质就是在两处同步一个Java对象(可以基于 JDK 本身的对象序列化或者基于 HTTP 协议的数据序列化)A 和 A!,但其中一个Java对象A进行方法调用时,通过RMI的代理能力转发给另一个Java对象A!进行执行,并将A!的执行结果作为A的执行结果。这样就实现了,运算流程的分布式计算。一般这两个对象存在于两台机器上。

  1. 将可以远程调用的对象进行序列化,然后绑定到RMI Server(被调方,运行者)中作为存根(stub)
  2. RMI Client 会先去下载stub反序列化然后发起client调用,RMI 底层(RMI Interface Layer & Transport Layer)会讲请求参数封装发送到RMI Server 
  3. RMI Server 接收到封装的参数,传递给桩(skeleton),由桩解析参数并且以参数调用对应的存根()stub方法。
  4. 存根方法在RMI Server执行完毕之后,返回结果将被RMI底层封装并传输给RMI Client(也就是主调方,调用者)

目前的Java版本已经不需要创建skeleton,也不需要rmic来编译stub了,但是我学习的时候还是使用的rmic编译的stub,所以示例也是这样做的。

RMI Server编写

编写RMI Server Interface,这个也会被用在客户端(RMI Client),供客户端使用。列出了开放远程调用的接口

 1 package org.lyh.server;
 2 
 3 import java.rmi.Remote;
 4 import java.rmi.RemoteException;
 5 
 6 /**
 7  * Created by lvyahui on 2016/4/22.
 8  */
 9 public interface ITimeServer extends Remote {
10     long getServerTime() throws RemoteException;
11     int add(int a,int b) throws RemoteException;
12 }

编写时间Server 接口

 1 package org.lyh.server.impl;
 2 
 3 import org.lyh.server.ITimeServer;
 4 
 5 import java.rmi.RemoteException;
 6 import java.rmi.server.RMIClientSocketFactory;
 7 import java.rmi.server.RMIServerSocketFactory;
 8 import java.rmi.server.UnicastRemoteObject;
 9 
10 /**
11  * Created by lvyahui on 2016/4/22.
12  */
13 public class TimeServer extends UnicastRemoteObject implements ITimeServer {
14 
15     public TimeServer(int port) throws RemoteException {
16         super(port);
17     }
18 
19     public TimeServer() throws RemoteException {
20     }
21 
22     public TimeServer(int port, RMIClientSocketFactory csf, RMIServerSocketFactory ssf) throws RemoteException {
23         super(port, csf, ssf);
24     }
25 
26     @Override
27     public long getServerTime() throws RemoteException {
28         return System.currentTimeMillis();
29     }
30 
31     @Override
32     public int add(int a, int b) throws RemoteException {
33         return a + b;
34     }
35 }

将编写好的Server实现对象绑定到Java RMI的名字服务上

 1 package org.lyh;
 2 
 3 import org.lyh.server.impl.TimeServer;
 4 
 5 import java.net.MalformedURLException;
 6 import java.rmi.Naming;
 7 import java.rmi.RemoteException;
 8 
 9 public class Main {
10     public static void main(String[] args) {
11         try {
12             TimeServer timeServer = new TimeServer();
13             /* 绑定到JVM 的 RMI Server上*/
14 //            Naming.bind("t1",timeServer);
15 //            Naming.rebind("t1",timeServer);
16             /* 当RMI注册server是指定了端口时或者不在本机运行时,需要这样写*/
17             Naming.rebind("//localhost/t1",timeServer);
18             System.out.println("Bind is Finish");
19         } catch (RemoteException e) {
20             e.printStackTrace();
21         } catch (MalformedURLException e) {
22             e.printStackTrace();
23         }
24     }
25 }

这时,要用rmic进一步编译TimeServer.class文件,得到TimeServer_stub.class 存根对象文件。如果不用rmic编译的方式,也可以通过写代码的方式获取stub。

因为我是使用IDEA开发的,所以我进入了RMI\out\production\RMI目录编译,执行rmic org.lyh.server.impl.TimeServer

 

这样会在相同目录下生成TimeServer_stub.class 文件

然后启动RMI名字注册服务,执行 rmiregistry 命令(jdk\bin下的一个脚本),可以执行 rmiregistry port执行端口,否则默认就是1099

下面可以执行org.lyh.Main@main方法了,将存根绑定(注册)到RMI 名字服务上,名字为t1

RMI Client编写

RMI client就简单了,拉取存根,然后发起调用,调用被传输到Server执行,并获取到执行结果,返回结果直接由接口方法return得到。

 1 package org.lyh.client;
 2 
 3 import org.lyh.server.ITimeServer;
 4 
 5 import java.net.MalformedURLException;
 6 import java.rmi.Naming;
 7 import java.rmi.NotBoundException;
 8 import java.rmi.RemoteException;
 9 
10 /**
11  * Created by lvyahui on 2016/4/22.
12  */
13 public class TimeClient {
14     public static void main(String[] args) {
15         try {
16            ITimeServer iTimeServer = (ITimeServer) Naming.lookup("rmi://192.168.18.1/t1");
17             System.out.println(iTimeServer.getServerTime());
18             System.out.println(iTimeServer.add(3,7));
19         } catch (NotBoundException e) {
20             e.printStackTrace();
21         } catch (MalformedURLException e) {
22             e.printStackTrace();
23         } catch (RemoteException e) {
24             e.printStackTrace();
25         }
26     }
27 }

为了更加真实,我将这个Client上传到虚拟机上编译运行

 

目录
相关文章
|
2月前
|
安全 前端开发 Java
《深入理解Spring》:现代Java开发的核心框架
Spring自2003年诞生以来,已成为Java企业级开发的基石,凭借IoC、AOP、声明式编程等核心特性,极大简化了开发复杂度。本系列将深入解析Spring框架核心原理及Spring Boot、Cloud、Security等生态组件,助力开发者构建高效、可扩展的应用体系。(238字)
|
5月前
|
人工智能 前端开发 Java
2025年WebStorm高效Java开发全指南:从配置到实战
WebStorm 2025不仅是一款强大的JavaScript IDE,也全面支持Java开发。本文详解其AI辅助编程、Java特性增强及性能优化,并提供环境配置、高效开发技巧与实战案例,助你打造流畅的全栈开发体验。
523 4
|
5月前
|
前端开发 JavaScript Java
Java 开发中 Swing 界面嵌入浏览器实现方法详解
摘要:Java中嵌入浏览器可通过多种技术实现:1) JCEF框架利用Chromium内核,适合复杂网页;2) JEditorPane组件支持简单HTML显示,但功能有限;3) DJNativeSwing-SWT可内嵌浏览器,需特定内核支持;4) JavaFX WebView结合Swing可完美支持现代网页技术。每种方案各有特点,开发者需根据项目需求选择合适方法,如JCEF适合高性能要求,JEditorPane适合简单展示。(149字)
613 1
|
5月前
|
安全 Java 领域建模
Java 17 探秘:不容错过的现代开发利器
Java 17 探秘:不容错过的现代开发利器
427 0
|
3月前
|
消息中间件 人工智能 Java
抖音微信爆款小游戏大全:免费休闲/竞技/益智/PHP+Java全筏开源开发
本文基于2025年最新行业数据,深入解析抖音/微信爆款小游戏的开发逻辑,重点讲解PHP+Java双引擎架构实战,涵盖技术选型、架构设计、性能优化与开源生态,提供完整开源工具链,助力开发者从理论到落地打造高留存、高并发的小游戏产品。
|
3月前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
156 0
|
4月前
|
JavaScript 安全 前端开发
Java开发:最新技术驱动的病人挂号系统实操指南与全流程操作技巧汇总
本文介绍基于Spring Boot 3.x、Vue 3等最新技术构建现代化病人挂号系统,涵盖技术选型、核心功能实现与部署方案,助力开发者快速搭建高效、安全的医疗挂号平台。
260 3
|
4月前
|
安全 Java 数据库
Java 项目实战病人挂号系统网站设计开发步骤及核心功能实现指南
本文介绍了基于Java的病人挂号系统网站的技术方案与应用实例,涵盖SSM与Spring Boot框架选型、数据库设计、功能模块划分及安全机制实现。系统支持患者在线注册、登录、挂号与预约,管理员可进行医院信息与排班管理。通过实际案例展示系统开发流程与核心代码实现,为Java Web医疗项目开发提供参考。
254 2