Rpc与RMI服务,springcloud视频教程

本文深入探讨RPC(远程过程调用)框架的基础能力,包括远程通信、CallID映射、序列化/反序列化和跨平台交互。介绍了Java中常见的RPC框架如Netty、bRPC、dubbo、gRPC和RMI,详细阐述了RMI的使用,包括RMI的三大基本类、服务端构建、注册表创建和客户端调用。同时,文章强调了学习RMI的思想和规范对于理解RPC的重要性。
摘要由CSDN通过智能技术生成

首先我们来思考一个问题,一个RPC框架至少拥有哪些能力?我们可以从下面几点来考虑:

远程通信能力

首先RPC得定义就是能实现网络端远程服务器之间的通信,所以RPC一定要有的最基础的能力,就是能发起远程的网络传输连接请求,即Sokect通讯的维护和协议能力封装

Call ID映射能力

RPC框架既然拥有跨服务端进行传输数据交互的能力,那么我们需要告知远程服务器如何调用Multiply ,要知道在本地调用中,函数体是直接通过函数指针来指定的,而在远程通信中,由于不在一个机器中,使用函数指针的方式是行不通的,所以在RPC框架中,几乎所有的函数都会有一个自己的ID,且保持唯一,服务端和客户端之间都会去维护ID在本机器中不同的地址,当有RPC请求传递过来的时候,服务端只需要查找本地维护的ID和地址的关系,找到对应的函数,进行调用,并且传输对应的结果即可

序列化/反序列化能力

当数据从远程服务器上返回,由于远程网络传输的原因,在Java中必须使用序列化标准来实现数据的传输,不能像本地调用一样简单的将内存数据读取,所以RPC框架一定要有序列化支持的能力,可以将远程的数据序列化为字节流,接受完毕后反序列化成对应的内存对象

跨平台/夸语言交互能力

由于RPC远程交互传输数据,这个时候服务端的系统环境以及所交互的进程使用的语言等并不是固定的,所以RPC框架需要拥有一个稳定的可以跨平台的,跨语言环境的传输协议支持

常见的RPC框架

了解了RPC框架最基础的能力以后,我们来看看Java开发中那些常见的经典RPC框架吧:

1.Netty:netty框架严格意义上并不属于RPC,更多的是作为一种网络协议框架,能够快速的提供高性能的RPC或者HTTP等的远程通信能力基础

2.bRPC:bRPC是一个基于protobuf接口的PRC框架,看到命名就可以知道此框架来自著名的百度团队,此框架囊括了百度内部的所有RPC协议,并且支持多种第三方协议,由于基于protobuf算法,从性能来看几乎是同类RPC框架中的领跑者

3.dubbo:dubbo框架是业界著名的阿里巴巴团队早期开源的优秀的RPC框架,此框架发展较成熟,能独立作为商业化组件使用,且依托于Spring使用

4.gRPC:gRPC是谷歌团队基于Netty开发实现的底层网络库,而此框架还有Go语言的版本,基于Net库开发

5.RMI:rmi可以说是java中最早的RPC框架之一,且此rpc框架由sun团队开发,集成于JDK中,可以说完全可以实现开箱即用,不需要任何外部jar依赖,但由于早期的rpc实现,且设计上并不是为了解决互联网企业类的高并发问题,所以不建议现在互联网开发中作为rpc实现

RMI框架基本使用

RMI既然是java团队设计出来的rpc框架,虽然现在已经不适合企业级生产使用,但是其中的思想和规范值得学习,我们就来看看RMI框架如何使用吧:

RMI三大基本类

实现RMI所需要的API基本都在三大类中,如下:

java.rmi:提供客户端需要的类、接口和异常;

java.rmi.server:提供服务端需要的类、接口和异常;

java.rmi.registry:提供注册表的创建以及查找和命名远程对象的类、接口和异常 ;

构建RMI服务端

首先在RMI中服务端供客户端调用的实例称之为远程对象,而在RMI中实现了java.rmi.Remote接口的类或者继承了java.rmi.Remote接口的都是RMI的远程对象。那么我们来定义一个接口,继承java.rmi.Remote

/**
 *用户处理器
**/
public interface UserHandler extends Remote {
    String getUserName(int id) throws RemoteException;
    String getUserPassWord() throws RemoteException;
    User getUserByName(String name) throws RemoteException;
}

这里需要注意的一点是,继承了Remote接口的接口中定义的所有的方法必须抛出RemoteException异常,并且该接口的实现类必须直接或者间接继承java.rmi.server.UnicastRemoteObject类,该类中提供了很多支持RMI的方法,可以通过JRMP协议导出一个远程对象的引用,生成动态代理构建的Stub对象,实现代码如下:

public class UserHandlerImpl extends UnicastRemoteObject implements UserHandler {
    //这里因为集继承了UnicastRemoteObject类,其构造器要抛出RemoteException,所以申明构造
    public UserHandlerImpl() throws RemoteException {
        super();
    }

    @Override
    public String getUserName(int id) throws RemoteException {
        return "pdc";
    }
    @Override
    public String getUserPassWord() throws RemoteException{
        return 654321;
    }
    @Override
    public User getUserByName(String name) throws RemoteException{
        return new User(name, 654321);
    }
}

这里我们构造了一个User实体,为了能实现远程传输,所以这里我们将其进行序列化:

public class User implements Serializable {
    private static final long serialVersionUID = 42L;
    
    private String name;
    private String passWord;

    public String getName(){
        return this.name;
    }
    
    public String getPassWord(){
        return this.passWord;
    }
    
    public void setName(String name){
        this.name = name;
    }
    
    public void setPassWord(String passWord){
        this.passWord = passWord;
    }
    
    public User(String name, String passWord) {
        this.name = name;
        this.passWord = passWord;
    }
}

需要注意的一点是,如果jdk版本低于1.5,需要手动运行rmic命令生成实现类的Stub对象,而1.5开始使用动态代理技术,已经可以自动生成Stub对象了,做完这些就可以启动服务端了:

UserHandler userHandler = null;
try {
    userHandler = new UserHandlerImpl();
    Naming.rebind("user", userHandler);//将当前的实例与名称为user绑定,后面客户端调用查找对应的名称
    System.out.println(" RMI 服务端启动成功");
} catch (Exception e) {
    System.err.println(" RMI 服务端启动失败");
    e.printStackTrace();
}

构建RMI注册表

其实所谓注册表就是保存了RMI服务端启动与绑定的名称的进程,由于jdk已经把RMI代码集成到了JDK中,RMI的注册表其实不需要写任何代码,在JDK的bin目录下已经存在一个叫rmiregistry.exe的程序,不过我们需要在当前的class类路径下启动注册表(所以需要注意JAVA_HOME环境变量一定要配置成功) ,来到class类路径下,输入命令:

rmiregistry 9999

即可指定rmi的注册表在9999端口中运行,如果不指定端口,默认使用1099,当然不想让RMI的注册表在前台显示,也可以输入后台运行命令:

start rmiregistry

构建RMI客户端

前面服务端和注册表都已经运行起来了,接下来我们需要的就是客户端发起访问的请求了,需要注意的是,User实例类和UserHandler接口在客户端代码中也有一份(企业开发过程中会依赖同一份代码),所以这里的客户端调用代码如下:

try {
    UserHandler handler = (UserHandler) Naming.lookup("user");//这里使用的user是服务端启动的时候绑定的名称
    String passWord = handler.getUserPassWord();
    String name = handler.getUserName(1);
    System.out.println("name: " + name);
    System.out.println("passWord: " + passWord);
    System.out.println("user: " + handler.getUserByName("pdc"));
} catch (Exception e) {
    e.printStackTrace();
}


## 结局:总结+分享

看完美团、字节、腾讯这三家的一二三面试问题,是不是感觉问的特别多,可能咱们真的又得开启面试造火箭、工作拧螺丝的模式去准备下一次的面试了。

开篇有提及我可是足足背下了**Java互联网工程师面试1000**,多少还是有点用的呢,换汤不换药,不管面试官怎么问你,抓住本质即可!能读到此处的都是真爱,所以福利也为你准备,这份1000题免费送你!**[点击这里免费领取](https://gitee.com/vip204888/java-p7)**

*   **Java互联网工程师面试1000**

![image.png](https://img-blog.csdnimg.cn/img_convert/971a908076810329926683e37761d89d.png)

而且从上面三家来看,算法与数据结构是必备不可少的呀,因此我建议大家可以去刷刷这本左程云大佬著作的 《程序员代码面试指南 IT名企算法与数据结构题目最优解》,里面近200道真实出现过的经典代码面试题。

*   **程序员代码面试指南--IT名企算法与数据结构题目最优解**

![image.png](https://img-blog.csdnimg.cn/img_convert/33e77993980f0a89f08cd88b23b8510f.png)


*   其余像设计模式,建议可以看看下面这4份PDF(已经整理)

![image.png](https://img-blog.csdnimg.cn/img_convert/f3b1778bb27cb5ea091bebfee9e491f1.png)


*   更多的Java面试学习笔记如下,关于面试这一块,我额外细分出Java基础-中级-高级开发的面试+解析,以及调优笔记等等等。。。

![image.png](https://img-blog.csdnimg.cn/img_convert/698d7b2db1562429f45795d13d4b90e9.png)


设计模式,建议可以看看下面这4份PDF(已经整理)

[外链图片转存中...(img-FSbMNxYW-1628588740637)]


*   更多的Java面试学习笔记如下,关于面试这一块,我额外细分出Java基础-中级-高级开发的面试+解析,以及调优笔记等等等。。。

[外链图片转存中...(img-gUsdKXRn-1628588740640)]


以上所提及的全部Java面试学习的PDF及笔记,如若皆是你所需要的,那么都可发送给你!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值