You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

16 KiB

31 | 动手实现一个简单的RPC框架原理和程序的结构

你好,我是李玥。

接下来的四节课我们会一起实现一个RPC框架。你可能会问为什么不实现一个消息队列而要实现一个RPC框架呢原因很简单我们课程的目的是希望你能够学以致用举一反三而不只是照猫画虎。在之前的课程中我们一直在讲解消息队列的原理和实现消息队列的各种技术那我们在实践篇如果再实现一个消息队列不过是把之前课程中的内容重复实现一遍意义不大。

消息队列和RPC框架是我们最常用的两种通信方式虽然这两种中间系统的功能不一样但是实现这两种中间件系统的过程中,有很多相似之处,比如,它们都是分布式系统,都需要解决应用间通信的问题,都需要解决序列化的问题等等。

实现RPC框架用到的大部分底层技术是和消息队列一样的也都是我们在之前的课程中讲过的。所以我们花四节课的时间来实现一个RPC框架既可以检验你对进阶篇中学习到的底层技术掌握的是不是扎实又可以学到RPC框架的实现原理买一送一很超值。

接下来的四节课我们是这样安排的。本节课我们先来学习RPC框架的实现原理然后我们一起看一下如何来使用这个RPC框架顺便给出整个项目的总体结构。第二节课中一起来实现RPC框架的通信与序列化部分最后的两节课分别来实现客户端与服务端这两部分。

下面我们先来一起了解一下RPC框架的实现原理。

首先需要明确一下RPC框架的范围。我们这里所说的RPC框架是指类似于Dubbo、gRPC这种框架使用这些框架应用程序可以“在客户端直接调用服务端方法就像调用本地方法一样。”而一些基于REST的远程调用框架虽然同样可以实现远程调用但它对使用者并不透明无论是服务端还是客户端都需要和HTTP协议打交道解析和封装HTTP请求和响应。这类框架并不能算是“RPC框架”。

RPC框架是怎么调用远程服务的

所有的RPC框架它们的总体结构和实现原理都是一样的。接下来我们以最常使用的Spring和Dubbo配合的微服务体系为例一起来看一下RPC框架到底是如何实现调用远程服务的。

一般来说,我们的客户端和服务端分别是这样的:

@Component
public class HelloClient {

    @Reference // dubbo注解
    private HelloService helloService;

    public String hello() {
      return helloService.hello("World");
    }
}

@Service // dubbo注解
@Component
public class HelloServiceImpl implements HelloService {

    @Override
    public String hello(String name) {
        return "Hello " + name;
    }
}

在客户端,我们可以通过@Reference注解获得一个实现了HelloServicer这个接口的对象我们的业务代码只要调用这个对象的方法就可以获得结果。对于客户端代码来说调用就是helloService这个本地对象但实际上真正的服务是在远程的服务端进程中实现的。

再来看服务端在服务端我们的实现类HelloServiceImpl实现了HelloService这个接口。然后我们通过@Service这个注解注意这个@Service是Dubbo提供的注解不是Spring提供的同名注解在Dubbo框架中注册了这个实现类HelloServiceImpl。在服务端我们只是提供了接口HelloService的实现并没有任何远程调用的实现代码。

对于业务代码来说无论是客户端还是服务端除了增加了两个注解以外和实现一个进程内调用没有任何区别。Dubbo看起来就像把服务端进程中的实现类“映射”到了客户端进程中一样。接下来我们一起来看一下Dubbo这类RPC框架是如何来实现调用远程服务的。

注意Dubbo的实现原理或者说是RPC框架的实现原理是各大厂面试中最容易问到的问题之一所以接下来的这一段非常重要。

在客户端业务代码得到的HelloService这个接口的实例并不是我们在服务端提供的真正的实现类HelloServiceImpl的一个实例。它实际上是由RPC框架提供的一个代理类的实例。这个代理类有一个专属的名称叫“桩Stub”。

在不同的RPC框架中这个桩的生成方式并不一样有些是在编译阶段生成的有些是在运行时动态生成的这个和编程语言的语言特性是密切相关的所以在不同的编程语言中有不同的实现这部分很复杂可以先不用过多关注。我们只需要知道这个桩它做了哪些事儿就可以了。

我们知道HelloService的桩同样要实现HelloServer接口客户端在调用HelloService的hello方法时实际上调用的是桩的hello方法在这个桩的hello方法里面它会构造一个请求这个请求就是一段数据结构请求中包含两个重要的信息

  1. 请求的服务名在我们这个例子中就是HelloService#hello(String)也就是说客户端调用的是HelloService的hello方法
  2. 请求的所有参数在我们这个例子中就只有一个参数name 它的值是“World”。

然后,它会把这个请求发送给服务端,等待服务的响应。这个时候,请求到达了服务端,然后我们来看服务端是怎么处理这个请求的。

服务端的RPC框架收到这个请求之后先把请求中的服务名解析出来然后根据这个服务名找一下在服务端进程中有没有这个服务名对应的服务提供者。

在这个例子的服务端中,由于我们已经通过@Service注解向RPC框架注册过HelloService的实现类所以RPC框架在收到请求后可以通过请求中的服务名找到HelloService真正的实现类HelloServiceImpl。找到实现类之后RPC框架会调用这个实现类的hello方法使用的参数值就是客户端发送过来的参数值。服务端的RPC框架在获得返回结果之后再将结果封装成响应返回给客户端。

客户端RPC框架的桩收到服务端的响应之后从响应中解析出返回值返回给客户端的调用方。这样就完成了一次远程调用。我把这个调用过程画成一张图放在下面你可以对着这张图再消化一下上面的流程。

在上面的这个调用流程中我们忽略了一个问题那就是客户端是如何找到服务端地址的呢在RPC框架中这部分的实现原理其实和消息队列的实现是完全一样的都是通过一个NamingService来解决的。

在RPC框架中这个NamingService一般称为注册中心。服务端的业务代码在向RPC框架中注册服务之后RPC框架就会把这个服务的名称和地址发布到注册中心上。客户端的桩在调用服务端之前会向注册中心请求服务端的地址请求的参数就是服务名称也就是我们上面例子中的方法签名HelloService#hello注册中心会返回提供这个服务的地址然后客户端再去请求服务端。

有些RPC框架比如gRPC是可以支持跨语言调用的。它的服务提供方和服务调用方是可以用不同的编程语言来实现的。比如我们可以用Python编写客户端用Go语言来编写服务端这两种语言开发的服务端和客户端仍然可以正常通信。这种支持跨语言调用的RPC框架的实现原理和普通的单语言的RPC框架并没有什么本质的不同。

我们可以再回顾一下上面那张调用的流程图,如果需要实现跨语言的调用,也就是说,图中的客户端进程和服务端进程是由两种不同的编程语言开发的。其实,只要客户端发出去的请求能被服务端正确解析,同样,服务端返回的响应,客户端也能正确解析,其他的步骤完全不用做任何改变,不就可以实现跨语言调用了吗?

在客户端和服务端收发请求响应的工作都是RPC框架来实现的所以**只要RPC框架保证在不同的编程语言中使用相同的序列化协议就可以实现跨语言的通信。**另外为了在不同的语言中能描述相同的服务定义也就是我们上面例子中的HelloService接口跨语言的RPC框架还需要提供一套描述服务的语言称为IDLInterface description language。所有的服务都需要用IDL定义再由RPC框架转换为特定编程语言的接口或者抽象类。这样就可以实现跨语言调用了。

讲到这里RPC框架的基本实现原理就很清楚了可以看到实现一个简单的RPC框架并不是很难这里面用到的绝大部分技术包括高性能网络传输、序列化和反序列化、服务路由的发现方法等都是我们在学习消息队列实现原理过程中讲过的知识。

下面我就一起来实现一个“麻雀虽小但五脏俱全”的RPC框架。

RPC框架的总体结构是什么样的

虽然我们这个RPC框架只是一个原型系统但它仍然有近50个源代码文件2000多行源代码。学习这样一个复杂的项目最好的方式还是先学习它的总体结构然后再深入到每一部分的实现细节中去所以我们一起先来看一下这个项目的总体结构。

我们采用Java语言来实现这个RPC框架。我们把RPC框架对外提供的所有服务定义在一个接口RpcAccessPoint中

/**
 * RPC框架对外提供的服务接口
 */
public interface RpcAccessPoint extends Closeable{
    /**
     * 客户端获取远程服务的引用
     * @param uri 远程服务地址
     * @param serviceClass 服务的接口类的Class
     * @param <T> 服务接口的类型
     * @return 远程服务引用
     */
    <T> T getRemoteService(URI uri, Class<T> serviceClass);

    /**
     * 服务端注册服务的实现实例
     * @param service 实现实例
     * @param serviceClass 服务的接口类的Class
     * @param <T> 服务接口的类型
     * @return 服务地址
     */
    <T> URI addServiceProvider(T service, Class<T> serviceClass);

    /**
     * 服务端启动RPC框架监听接口开始提供远程服务。
     * @return 服务实例,用于程序停止的时候安全关闭服务。
     */
    Closeable startServer() throws Exception;
}

这个接口主要的方法就只有两个第一个方法getRemoteService供客户端来使用这个方法的作用和我们上面例子中Dubbo的@Reference注解是一样的客户端调用这个方法可以获得远程服务的实例。第二个方法addServiceProvider供服务端来使用这个方法的作用和Dubbo的@Service注解是一样的服务端通过调用这个方法来注册服务的实现。方法startServer和close在父接口Closeable中定义用于服务端启动和停止服务。

另外我们还需要定一个注册中心的接口NameService

/**
 * 注册中心
 */
public interface NameService {
    /**
     * 注册服务
     * @param serviceName 服务名称
     * @param uri 服务地址
     */
    void registerService(String serviceName, URI uri) throws IOException;

    /**
     * 查询服务地址
     * @param serviceName 服务名称
     * @return 服务地址
     */
    URI lookupService(String serviceName) throws IOException;
}

这个注册中心只有两个方法分别是注册服务地址registerService和查询服务地址lookupService。

以上就是我们要实现的这个RPC框架的全部功能了。然后我们通过一个例子看一下这个RPC框架如何来使用。同样需要先定义一个服务接口

public interface HelloService {
    String hello(String name);
}

接口定义和本节课开始的例子是一样的。然后我们分别看一下服务端和客户端是如何使用这个RPC框架的。

客户端:

URI uri = nameService.lookupService(serviceName);
HelloService helloService = rpcAccessPoint.getRemoteService(uri, HelloService.class);
String response = helloService.hello(name);
logger.info("收到响应: {}.", response);

客户端首先调用注册中心NameService的lookupService方法查询服务地址然后调用rpcAccessPoint的getRemoteService方法获得远程服务的本地实例也就是我们刚刚讲的“桩”helloService。最后调用helloService的hello方法获得返回值并打印出来。

然后来看服务端首先我们需要有一个HelloService的实现

public class HelloServiceImpl implements HelloService {
    @Override
    public String hello(String name) {
        String ret = "Hello, " + name;
        return ret;
    }
}

然后我们将这个实现注册到RPC框架上并启动RPC服务

rpcAccessPoint.startServer();
URI uri = rpcAccessPoint.addServiceProvider(helloService, HelloService.class);
nameService.registerService(serviceName, uri);

首先启动RPC框架的服务然后调用rpcAccessPoint.addServiceProvider方法注册helloService服务然后我们再调用nameServer.registerService方法在注册中心注册服务的地址。

可以看到我们将要实现的这个RPC框架的使用方式总体上和上面使用Dubbo和Spring的例子是一样的唯一的一点区别是由于我们没有使用Spring和注解所以需要用代码的方式实现同样的功能。

我把这个RPC框架的实现代码以及上面如何使用这个RPC框架的例子放在了GitHub的simple-rpc-framework项目中。整个项目分为如下5个Module

其中RPC框架提供的服务RpcAccessPoint和注册中心服务NameService这两个接口的定义在Module rpc-api中。使用框架的例子HelloService接口定义在Module hello-service-api中例子中的客户端和服务端分别在client和server这两个Module中。

后面的三节课我们将一起来实现这个RPC框架也就是Module rpc-netty。

小结

从这节课开始我们要用四节课利用之前学习的、实现消息队列用到的知识来实现一个RPC框架。

我们在实现RPC框架之前需要先掌握RPC框架的实现原理。在RPC框架中最关键的就是理解“桩”的实现原理桩是RPC框架在客户端的服务代理它和远程服务具有相同的方法签名或者说是实现了相同的接口。客户端在调用RPC框架提供的服务时实际调用的就是“桩”提供的方法在桩的实现方法中它会发请求的服务名和参数到服务端服务端的RPC框架收到请求后解析出服务名和参数后调用在RPC框架中注册的“真正的服务提供者”然后将结果返回给客户端。

思考题

课后你需要从GitHub上把我们即将实现的RPC框架的源代码下载到本地先分别运行一下例子中的服务端和客户端对整个项目有一个感性的认识。然后再分别看一下rpc-api、hello-service-api、server和client这四个Module的源代码理清楚RPC框架的功能以及如何使用这个RPC框架为后续三节课的学习做好准备。欢迎在留言区与我分享讨论。

感谢阅读,如果你觉得这篇文章对你有一些启发,也欢迎把它分享给你的朋友。