实现一个迷你版的RPC

作者: 张丰哲 | 来源:发表于2017-11-17 22:31 被阅读2367次

    前言

    在实际后台服务开发中,比如订单服务(开发者A负责)需要调用商品服务(开发者B负责),那么开发者B会和A约定调用API,以接口的形式提供给A。通常都是B把API上传到Maven私服,然后B开始写API的实现,A只需要引入API依赖进行开发即可。

    订单服务调用商品服务

    上图简单的描述了RPC在实际场景中的应用,我们在开发中当然是利用现有的RPC框架来快速实现业务需求,比如百度开源了baidu-rpc,阿里的Dubbo早已声名在外,腾讯自己玩TAF。本篇博客将实现一个迷你版的RPC,探索下RPC底层实现的奥秘!


    动手实现RPC

    商品服务工程

    商品服务工程

    注意,我将商品服务的API以及实现分为Maven的2个模块来开发。这里,我们想给定一个商品ID,查询得到商品对象信息。

    商品对象

    商品API Product

    要注意的是,Product是可以被序列化的,Why?

    很显然,订单系统调用商品系统的时候,需要商品系统返回一个商品,必然涉及到发生网络传输,这就涉及对象的序列化和反序列化了。

    商品查询API接口

    商品查询API

    订单系统调用商品服务

    订单服务调用商品服务

    在订单系统工程中需要引入商品服务API依赖。

    在上图代码中,最重要的就是rpc方法了!

    rpc实现方法

    rpc

    第一,我们看到了Proxy.newProxyInstance,很显然在进行动态代理。也即是说,在订单服务调用商品服务的代码中,我们先是通过动态代理返回一个代理的IProductService类型对象,这意味着当代理对象调用queryById方法的时候,会自动调用invoke方法!

    第二,我们看看invoke到底做了些什么?

    它本质上就是进行Socket通信,那么它需要传递什么信息给到商品服务呢?

    我们知道订单系统就是想调用商品服务的某个类的某个方法,然后把这个方法的返回结果传输给订单系统!

    想一想,如何调用某个类的某个方法呢?

    只要我们能确定这个类的全限定类名、确定方法名、确定方法的参数类型,给定方法需要的具体参数,通过反射就能实现。

    商品服务调用后得到的结果,我们序列化写入Socket流中,在订单系统中反序列化得到对象即可。

    第三,这里需要思考一个问题:在订单系统中我们只知道商品服务的API,并不知道这背后的API到底是如何实现的,所以我们需要有一个映射,就是商品服务的API到商品服务的实现的一个映射关系,其实这就是所谓的服务的注册!

    商品API的具体实现

    商品服务 商品服务API实现

    商品服务

    商品服务

    从这里,可以清晰的看到,商品服务读取了订单系统调用商品系统时发送的数据,利用反射机制,进行方法调用,并把调用结果写入Socket输出流。

    运行结果

    运行结果

    启动商品服务后,通过订单系统发起对商品服务的调用。

    以前总认为RPC是遥不可及的,感觉是个很神奇的东西,实际上它的底层实现不就是这样的么~

    晚安!

    2017.11.17 by zhangfengzhe

    相关文章

      网友评论

      • kndsk:客户端得定义和服务端一样的api才能调用吧?
        kndsk:@张丰哲 非常感谢:smile::smile:
        张丰哲:@kndsk 客户端需要得到服务端api的接口即可,可以先服务端定义api接口后上传到maven,然后客户端依赖就可以开始各自开发了哈。
      • 266c7b31c70f:在订单系统中我们只知道商品服务的API,并不知道这背后的API到底是如何实现的,如果有多个实现怎么办?什么时候把代码贴下:smile:
        张丰哲:如文章所述,这是服务注册的概念,一般可以通过Zookeeper来实现。代码的话,以后我改进下,提供下载链接吧:smile:
      • f9d3d9cdfec1:这样写是不是服务提供方每次提供服务都会new 一个新的对象
        张丰哲:恩恩,博文的代码中服务提供者每次都会反射new一个对象进行调用。
      • f9d3d9cdfec1:写的很好,浅显易懂
        张丰哲:欢迎继续关注,会继续更新的~
      • 逍遥哥_d879:rpc就是双方连接到socket,收到指令执行吧
      • 秋露凋枫:您好,可以提供一下源码吗
        张丰哲:可以,等我整理下,在贴出来地址吧。
      • iHelin:您好,请问实际的rpc框架真的是使用网络、代理和反射吗?感觉这些步骤都是比较耗时的。
        张丰哲:实际的RPC框架粗线条来说,是这样的,:smile:
      • 江江的大猪:为啥同一个demo用了eclipse和idea两个IDE:joy:
        张丰哲:主要是代码截图方便,一个是idea,一个是sublime
      • 莫那一鲁道:可以可以,底层原理都很简单。只不过大佬们为了性能和扩展性将框架变得更加的复杂
      • 失落随城:表示学了3年编程只看到了点皮毛!后悔虚度光阴呀
        张丰哲:努力什么时候都不晚,加油~

      本文标题:实现一个迷你版的RPC

      本文链接:https://www.haomeiwen.com/subject/vcpnmxtx.html