Bootstrap

RPC框架

RPC框架原理

RPC(remote procedure call,远程过程调用),一般用来实现部署在不同机器上的系统之间的方法调用,使得程序能够像访问本地资源一样,通过网络传输去访问远端系统资源。

RPC框架实现原理如下图:
在这里插入图片描述

Client Code:客户端调用方代码实现,负责发起RPC调用,为调用方用户提供使用API。
Seriaization/Deserialization:负责对RPC调用通过网络传输的内容进行序列化和反序列化,不同的RPC框架有不同的实现机制。主要分为文本和二进制两大类。文本类别的序列化机制主要有XML和JSON两种格式,二进制类别的序列化机制常见的有Java原生的序列化机制,以及Hessian、protobuf、Thrift、Avro、Kryo、MessagePack等,不同的序列化方式在可读性、码流大小、支持的数据类型及性能等方面都存在较大差异,需要用户根据自己的实际情况进行甄别与筛选。
Stub Proxy:可以看作一种代理对象,屏蔽RPC调用过程中复杂的网络处理逻辑,使RPC调用透明化,能够保持与本地调用一样的代码风格。
Transport:作为RPC框架底层的通信传输模块,一般通过Socket在客户端与服务器之间传递请求与应答消息。
Server Code:服务端服务业务逻辑具体的实现。

RMI

什么是RMI

Java RMI(Remote Method Invocation)是一种基于Java的远程方法调用技术,是Java特有的一种RPC实现。它能够使部署在不同主机上的Java对象之间进行透明的通信与方法调用。

RMI调用示意图:
在这里插入图片描述

原生RMI代码示例

定义RMI对外服务接口HelloService:

import java.rmi.Remote;
import java.rmi.RemoteException;

public interface HelloService extends Remote{
    String sayHello(String someOne)throws RemoteException;
}

RMI接口方法定义必须显示声明抛出RemoteException异常。

服务端接口实现HelloServiceImpl:

import java.rmi.server.UnicastRemoteObject;
import java.rmi.RemoteException;
public class HelloServiceImpl extends UnicastRemoteObject implements HelloService {
private static final long serialVersionUID=-6190513770400890033L;
public HelloServiceImpl() throws RemoteException{
    super();
}
    @Override
    public String sayHello(String someOne) throws RemoteException {
        return "Hello,"+someOne;
    }
}

服务端方法实现必须继承UnicastRemoteObject类,该类定义了服务调用方与服务提供方对象示例,并建立一对一的连接。

服务端RMI服务启动代码:

import java.rmi.Naming;
import java.rmi.registry.LocateRegistry;

public class ServerMain {
    public static void main(String[] args)throws Exception {
        //创建服务
        HelloService helloService=new HelloServiceImpl();
        //注册服务
        LocateRegistry.createRegistry(8801);
        Naming.bind("rmi://localhost:8801/helloService",helloService);
        System.out.println("ServerMain provide RPC service now");
    }
}

客户端远程调用RMI服务代码:

import java.rmi.Naming;

public class ClientMain {
    public static void main(String[] args)throws Exception {
        //服务引入
        HelloService helloService=(HelloService) Naming.lookup("rmi://localhost:8801/helloService");
        //调用远程方法
        System.out.println("RMI 服务器返回的结构是:"+helloService.sayHello("helu"));
    }
}

首先运行服务端程序ServerMain,然后运行客户端ClientMain,运行结果:
在这里插入图片描述
问题1:各个序列化格式之间有什么区别?

;