java远程调用之RMI(终于可以自己写代码控制别人电脑了)

简介: 之前在研究生课程当中学了分布式系统这门课,而且还是自己的导师讲的这门课,在课堂上迷迷糊糊的晃悠了一学期,除了听见几个名词,也没太多印象。正好这几天用到远程过程调用,使用的是gRPC,想到之前上课听过,于是把这块的知识从书到教程好好地补充了一下。本篇文章尽量不啰嗦,它的重要性自己可以私下了解一下。

一、什么是RMI


1、概念


RMI又叫做远程方法调用。在java中表示一台java虚拟机上的程序调用另外一台java虚拟机上的程序。QQ里面有一个远程控制别人电脑,其实就是基于这个原理。这一篇只是讲解一些基础的使用,如果深入进去,你也可以写一个远程控制器,去控制别人的电脑。


2、原理


既然是一个java虚拟机上的程序调用另外一台java虚拟机上的程序,那肯定是要经过通信的,通信的话就要遵循通信协议,使用通信方法。java提供了Socket来进行通信,封装了底层的通信协议。于是RMI就是通过socket进行通信的。我们画一张图来看一下整个RMI调用的流程。

v2-3c865d4ba5d56d98dbf0875ab66d594a_1440w.jpg就是整个流程,我是使用画图工具画的,不好看还请见谅。这里面涉及到了四个名词client、server、Stub、Skeleton。我们先对这4个名词先进性解释一下,然后再来看整个流程。


(1)名词介绍


第一个:client和server

意思就是客户端和服务器嘛,也就是一个调用方法者,一个被调用者,这个很好理解。


第二个:Stub和Skeleton

想要理解这俩名词的含义,我们可以举个例子,比如说我们想要找一位大老板帮忙,首先我们不直接去,找一个代理,这个代理先找到大老板的管家,通过管家再认识大老板。我们可以画一张图。

v2-c0af3e7aee6135129a19b1c21bd763f9_1440w.jpg

上面就是涉及到的几个名词含义。很容易理解。下面我们就真正开始介绍一下流程。


(2)RMI流程


第一步:客户端请求代理

第二步:Stub编码处理消息

第三步:消息传输

第四步:到达管家skeleton并处理信息

第五步:管家skeleton把信息提交给server

第六步:server接收到请求

第七步:server把请求的结果给管家

第八步:管家skeleton把结果转交给stub

第九步:代理Stub对结果解码

第十步:Stub把解码的结果交给client。

以上就是RMI的原理和流程。


二、代码实现


以上我们把RMI的概念以及原理介绍完了,最主要的还是如何去使用。使用的时候也有一个开发流程。


1、编写server接口,继承Remote。 2、编写server接口实现类,继承UnicastRemoteObject 类。 3、server创建一个注册表,指定端口。 4、server往注册表中注册server服务。 5、client获取注册表调用远程server方法。

这就是整个基本开发流程。在这里我们举个例子,客户端想要查询天气。调用某个城市,server就能返回某个城市的天气信息。(你可以在两台电脑上编写代码,更能体会到其思想)


1、创建server接口


public interface WeatherService extends Remote{
    //提供查询天气的功能
    //必须要继承Remote和抛出RemoteException异常
    String findWeather(String city) throws RemoteException;
}

2、创建server接口实现类

public class ServiceImpl extends UnicastRemoteObject implements WeatherService{
    //必须要有构造方法
    protected ServiceImpl() throws RemoteException {
        super();
    }
    //提供查询天气的功能:只提供西安和上海
    @Override
    public String findWeather(String city) throws RemoteException {
        if(city.equals("西安")) {
            return city+":多云,有大风,多穿衣服保暖";
        }
        else if(city.equals("上海")) {
            return city+":晴,温度40度,少穿衣服";
        }
        return "输入有误,请重新输入";
    }
}

在这里只提供了两个城市,真实情况下肯定不是这样的。需要到数据库里面获取天气信息。


3、server创建注册表并注册服务

public class RMIServer {
    public static void main(String[] args) {
        try {
            //1、创建一个服务
            WeatherService weatherService = new ServiceImpl();
            //2、生成注册表
            Registry registry = LocateRegistry.createRegistry(8000);
            //4、把代理绑定到注册表上发布:weather是暗号
            Naming.rebind("rmi://10.70.170.53:8000/weather",weatherService);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

4、创建client,获取注册表并调用服务

public class RMIClient {
    public static void main(String[] args) {
        try {
            //1、获取注册表
            WeatherService weatherService = null;
            try {
                weatherService = 
                (WeatherService) Naming.lookup("rmi://10.70.170.53:8000/weather");
            } catch (MalformedURLException e) {
                e.printStackTrace();
            } 
            //2、调用服务
            String xian=weatherService.findWeather("安");
            System.out.println(xian);
        } catch (RemoteException | NotBoundException e) {
            e.printStackTrace();
        }
    }
}

这就是整个基本流程,其实如果我们学过webservice的话应该都会很熟悉这个过程,不过webservice更加的麻烦。下面我们验证一下。首先输入只西安的“安”发现没有就返回错误,然后再输入西安”修正,就返回正确结果了。


三、总结


RMI的用途可以说是超级广泛了,比如说Dubbo、Spring。具体如何使用这里就不再描述了,不过java提供的RMI还是很少用到的,如果我们知道它的原理,可以完成一些在RMI之上的一些功能。比如说远程控制电脑

相关文章
|
15天前
|
Java 测试技术 应用服务中间件
常见 Java 代码缺陷及规避方式(下)
常见 Java 代码缺陷及规避方式(下)
43 0
|
17天前
|
Java
Java中ReentrantLock释放锁代码解析
Java中ReentrantLock释放锁代码解析
25 8
|
20天前
|
前端开发 小程序 Java
uniapp上传图片 前端以及java后端代码实现
uniapp上传图片 前端以及java后端代码实现
33 0
|
22天前
|
设计模式 存储 Java
23种设计模式,享元模式的概念优缺点以及JAVA代码举例
【4月更文挑战第6天】享元模式(Flyweight Pattern)是一种结构型设计模式,旨在通过共享技术有效地支持大量细粒度对象的重用。这个模式在处理大量对象时非常有用,特别是当这些对象中的许多实例实际上可以共享相同的状态时,从而可以减少内存占用,提高程序效率
35 4
|
22天前
|
设计模式 Java 中间件
23种设计模式,适配器模式的概念优缺点以及JAVA代码举例
【4月更文挑战第6天】适配器模式(Adapter Pattern)是一种结构型设计模式,它的主要目标是让原本由于接口不匹配而不能一起工作的类可以一起工作。适配器模式主要有两种形式:类适配器和对象适配器。类适配器模式通过继承来实现适配,而对象适配器模式则通过组合来实现
31 4
|
23天前
|
存储 缓存 算法
优化 Java 后台代码的关键要点
【4月更文挑战第5天】本文探讨了优化 Java 后台代码的关键点,包括选用合适的数据结构与算法、减少不必要的对象创建、利用 Java 8 新特性、并发与多线程处理、数据库和缓存优化、代码分析与性能调优、避免阻塞调用、JVM 调优以及精简第三方库。通过这些方法,开发者可以提高系统性能、降低资源消耗,提升用户体验并减少运营成本。
|
24天前
|
Java 开发工具 流计算
flink最新master代码编译出现Java Runtime Environment 问题
在尝试编译Flink源码时遇到Java运行时环境致命错误:EXCEPTION_ACCESS_VIOLATION。问题出现在JVM.dll+0x88212。使用的是Java 11.0.28和Java HotSpot(TM) 64-Bit Server VM。系统为Windows客户端,没有生成核心dump文件。错误日志保存在hs_err_pid39364.log和replay_pid39364.log。要解决这个问题,建议检查JDK版本兼容性,更新JDK或参照错误报告文件提交Bug至http://bugreport.java.com/bugreport/crash.jsp。
|
25天前
|
Java
使用Java代码打印log日志
使用Java代码打印log日志
82 1
|
25天前
|
设计模式 Java 数据库
Java设计模式精讲:让代码更优雅、更可维护
【4月更文挑战第2天】**设计模式是解决软件设计问题的成熟方案,分为创建型、结构型和行为型。Java中的单例模式确保类仅有一个实例,工厂方法模式让子类决定实例化哪个类。适配器模式则协调不兼容接口间的合作。观察者模式实现了一对多依赖,状态变化时自动通知相关对象。学习和适当应用设计模式能提升代码质量和可维护性,但需避免过度使用。设计模式的掌握源于实践与不断学习。**
Java设计模式精讲:让代码更优雅、更可维护
|
26天前
|
SQL 设计模式 安全
Java单例模式几种写法以及代码案例拿来直接使用
Java单例模式几种写法以及代码案例拿来直接使用
32 0