马上观RPC

马上观RPC


RPC是Remote Procedure Call的缩写,即远程过程调用。RPC是一系列技术的统称,通过这些技术,可以让应用程序像调用本地服务一样调用远程服务。与RPC类似的概念还有IPC和LPC。IPC即Inter-Process Comunication(进程间通信),LPC即Local Procedure Call(本地过程调用)。这些概念在传播中不免有交叉混用的情况。

本文不是对RPC等术语做严格定义,不会对相关技术做深入的分析。本文更像一个对RPC相关知识的不完全概括,和对RPC部分相关技术的简介。本文是近期对RPC快餐式学习的总结。

常用的IPC机制

IPC是指进程间通信,包含一系列技术。虽然RPC也可以算是进程间通信,但IPC更多是说基于操作系统内核提供的一些机制,让运行在同一个设备上的不同进程之间交换数据的一系列技术的统称。

大家提到IPC的时候,主要指代以基本数据和信号传输为主的通用通信机制。在应用程序中IPC基本处于底层数据通道的地位。不同操作系统提供的IPC机制略有不同。下面对基础的IPC机制做个不完全的列举。

Linux:

  • 管道(Pipe):匿名管道和命名管道;
  • 信号量(Semophore);
  • 信号(Signal);
  • 内存映射(mmap);
  • 套接字(Socket);

Windows:

  • 管道,同Linux;
  • 邮槽(Mail Slot);
  • 动态数据交换(DDE);
  • 文件映射(File Mapping);
  • 窗口事件;
    • 特殊的事件WM_COPYDATA:系统内核会负责跨进程内存拷贝;
  • 剪贴板;
  • DLL共享内存段;
  • 套接字(Socket);

Android(以JVM可用的为主):

  • 共享内存/内存映射(MemoryFile/mmap);
  • 基于binder驱动的若干机制:
    • Intent和四大组件;
    • Binderaidl
    • 基于File Descriptor的文件共享;

RPC的结构

RPC即远程过程调用。虽然在同一个设备上运行的不同进程间的调用也时不时被称作RPC,但RPC更多的是以跨设备调用为默认语境,即所谓的“远程”。话又说回来,有的RPC机制支持以系统内的IPC机制为底层通信协议实现设备内的高效调用,所以把同设备内的调用说成RPC也不算全错。

除了“远程”以外,“调用”的外观也是RPC的一个特性。所谓调用,是指程序可以像调用本地代码一样调用位于其他进程或设备上的代码。RPC不只是单纯的数据通信通道,而是在通信通道之上建立一套体系,把数据通信的过程对开发者隐藏起来。

下图是RPC的基本结构:

RPC架构.png

RPC的通信协议

所有RPC都是基于一套或多套通信协议作为远程通信通道。有的RPC机制只支持特定协议,甚至对特定协议有较强耦合;有的则可以支持在多个协议中任意选择搭配,其中也有以操作系统的IPC为协议,实现系统内进程间调用的例子。

不同RPC可能会选择不同的协议。例如:TCP、UDP、HTTP、protobuf、mqtt、等,甚至私有协议。RPC可能会基于下面的考虑选择协议:

  1. 网络连接的维持和QoS;
  2. 数据传输的安全性;
  3. 数据传输的体积和效率;
  4. 设备和网络环境对协议的兼容性和限制;
  5. 协议对自定义数据结构的易扩展性;
  6. 等等。

过程调用和Stub

通信协议只负责无差别二进制数据的传输,而RPC要过程调用的界面,协议和调用界面的差距,就是由Stub(存根)来填补的。

以一次调用过程为例,结合上面的架构图,介绍Stub的职责(这里先省略接口注册环节):

  1. Client本地代码调用被Client Stub代理的sample过程,代理的sample过程跟远程代码具有相同的函数原型;
  2. Client Stub把函数调用翻译成特定格式的二进制流,其中包含函数标识和序列化的所有入参;
  3. Client Stub把二进制流通过通信协议发送到Server端;
  4. Server Stub从通信协议接收二进制流,从中解析出目标函数标识,并反序列化所有入参;
  5. Server Stub根据上一步解析的数据,调用目标sample过程;
  6. Server Stub把本地sample过程的返回数据按照跟上面步骤相反的过程,回传给Client Stub;
  7. Client Stub把收到的返回值返回给本地调用代码。

通过上述步骤,Stub对上层代码隐藏了远程调用和本地调用的差异。

接口定义和IDL

上一节介绍了Stub代理调用和隐藏细节的作用。在RPC中Stub代码通常不用人工实现,而是使用一些工具自动生成。Stub代码生成工具通常以IDL文件作为输入。

IDL即中间描述语言,是一系列与具体编程语言无严格关系的语言。IDL的作用是描述远程接口,其中包含接口的原型,以及生成Stub代码所需的其他信息。IDL和代码生成工具通常由RPC框架定义和提供。

下面是Microsoft RPC的MIDL示例:

[ 
    uuid (ba209999-0c6c-11d2-97cf-00c04f8eea45), 
    version(1.0)
] 
interface math
{
    int Add([in] int x, [in] int y);
}

下面Android AIDL的示例:

package com.example;

interface ISampleAidl {
    void sendFile(in ParcelFileDescriptor fd);
}

接口注册和发现

在过程调用之前,需要Client端跟Server端建立某种连接,以确保端到端的数据通道。建立连接需要一个接口注册中心提供Server的位置(如IP地址、域名、等)和监听信息(如端口等),这些信息通过Server端的接口注册提供。

担任接口注册中心的,可能是一台服务器,也可能是Server端的某个进程。不同RPC框架可能会做出不同的设计。

使用RPC的基本步骤

使用RPC一般会包含下面的步骤:

  1. 用IDL定义接口及相关信息;
  2. 使用IDL工具生成Stub代码;
  3. Server端实现接口;
  4. Server端向注册中心注册接口和监听信息;
  5. Client端发现和连接Server;
  6. Client端调用Client Stub;

有的RPC框架可能会省略或自动完成部分步骤。

Android基于Binder提供的IPC机制

Android通过AIDL定义接口,例如:

// IFileBinder.aidl
interface IFileBinder {
    ParcelFileDescriptor openFileDescriptor(int mode);
}

android-gradle-plugin会在编译时生成Stub代码。Android从Stub中细分出了Stub/Proxy结构,其中Server端实现接口的时候需要继承自Stub类,而Proxy本质是Client Stub,供Client端代码调用。

// IFileBinder.java(已省略大量代码,只保留了关键代码)
public interface IFileBinder extends android.os.IInterface {
    /** Local-side IPC implementation stub class. */
    public static abstract class Stub extends android.os.Binder implements IFileBinder {
        public static phantom.demo.fileprovider.common.IFileBinder asInterface(android.os.IBinder obj) {
            //...
            return new IFileBinder.Stub.Proxy(obj);
        }
        private static class Proxy implements IFileBinder {
            //...
        }
    }
}
// Server端的接口实现
public class FileBinder extends IFileBinder.Stub {
    //...
}
// Client端调用
IBinder binder = ...;
int mode = ...;
IFileBinder fileBinder = IFileBinder.Stub.asInterface(binder); // 返回的是Proxy
ParcelFileDescriptor fd = fileBinder.openFileDescriptor(mode);

Android提供了一个binder驱动作为系统内大量IPC的基础。在framework中继续binder驱动封装了Binder类,相当于Stub代码的通信协议。

// Server端(已省略大量代码,只保留了关键代码)
public static abstract class Stub extends android.os.Binder implements IFileBinder {
    @Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException {
        _arg0 = data.readInt();
        android.os.ParcelFileDescriptor _result = this.openFileDescriptor(_arg0);
        _result.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
        return true;
    }
}
// Client端(已省略大量代码,只保留了关键代码)
private static class Proxy implements IFileBinder {
    @Override public ParcelFileDescriptor openFileDescriptor(int mode) throws RemoteException {
        mRemote.transact(Stub.TRANSACTION_openFileDescriptor, _data, _reply, 0);
        _result = android.os.ParcelFileDescriptor.CREATOR.createFromParcel(_reply);
        return _result;
    }
}

Android基于binder驱动实现了一个系统服务ServiceManager。ServiceManager就是注册中心,通过名称注册和发现接口。关于ServiceManager这里不做更多展开。

Android通过Stub类和Binder类自动完成了接口注册和接口发现,并通过相关系统接口自动完成了从Client端到Server端的连接。关于Binder机制的细节这里不做更多展开。

public static abstract class Stub extends android.os.Binder implements IFileBinder {
    private static final java.lang.String DESCRIPTOR = "com.example.IFileBinder";
    /** Construct the stub at attach it to the interface. */
    public Stub() {
        this.attachInterface(this, DESCRIPTOR);
    }
}

关于AIDL使用方面的详细介绍,可以参考官方文档Android 接口定义语言 (AIDL)

Windows提供的RPC机制

Microsoft RPC使用MIDL来定义接口,接口文件以.idl为扩展名。开发者可以用uuidgen工具生成接口的UUID并生成.idl文件模板。

uuidgen -i -oMyApp.idl

.idl文件如下:

[
  uuid(ba209999-0c6c-11d2-97cf-00c04f8eea45),
  version(1.0)
]
interface INTERFACENAME
{
}

开发者需要把INTERFACENAME替换为接口的实际名称,并添加接口方法。

使用midl命令行工具编译.idl文件可以得到Stub代码:

midl MyApp.idl

Stub代码是C程序文件,包含接口头文件MyApp.h、Server Stub实现MyApp_s.c、和Client Stub实现MyApp_c.c

MyApp.hMyApp_s.c需要参与Server端程序的编译和链接,Server端代码按照MyApp.h的声明实现接口。MyApp.hMyApp_c.c需要参与Client端程序的编译和连接,Client端代码按照MyApp.h的声明调用接口。

Microsoft RPC的注册中心叫终结点映射(endpoint map),它运行在Server主机中。

Server端需要通过如下步骤完成接口注册(下面是关键代码示例):

RPC_STATUS status;
// 1. 注册接口
status = RpcServerRegisterIf(MyInterface_v1_0_s_ifspec, NULL, NULL);
// 2. 绑定协议
status = RpcServerUseAllProtseq(L"ncacn_ip_tcp", RPC_C_PROTSEQ_MAX_REQS_DEFAULT, NULL);
RPC_BINDING_VECTOR *rpcBindingVector;
status = RpcServerInqBindings(&rpcBindingVector);
// 3. 注册endpoint
status = RpcEpRegister(MyInterface_v1_0_s_ifspec, rpcBindingVector, NULL, NULL);
// 4. 开始监听Client请求
status = RpcServerListen(1, RPC_C_LISTEN_MAX_CALLS_DEFAULT, 0);

Client端需要通过如下步骤完成接口发现和建立连接(下面是关键代码示例):

RPC_STATUS status;
// 1. 指定通信协议和Server端域名
unsigned short *StringBinding;
status = RpcStringBindingCompose(NULL, L"ncacn_ip_tcp", L"www.example.com", NULL, NULL, &StringBinding);
// 2. 创建绑定句柄
RPC_BINDING_HANDLE BindingHandle;
status = RpcBindingFromStringBinding(StringBinding, &BindingHandle);

Client完成上述步骤,就可以调用接口了。

Server端和Client端除了要分别执行上述步骤外,还需要为RPC提供MIDL内存管理,例如:

void __RPC_FAR * __RPC_USER midl_user_allocate(size_t len) {
    return(malloc(len));
}

void __RPC_USER midl_user_free(void __RPC_FAR * ptr) {
    free(ptr);
}

Microsoft RPC支持多种通信协议,开发者可以指定一个或多个协议,然后交给系统来选择。可选的协议包括但不限于:

  • TCP(如上面的示例代码)
  • UDP
  • HTTP
  • Named Pipe(命名管道)
  • LPC

命名管道和LPC可用于系统内的进程间调用。Windows的LPC是非公开接口,主要供系统服务使用,应用程序不能直接调用LPC接口。关于协议的清单可以参考Protocol Sequence Constants

关于Microsoft MIDL的更多介绍,可以参考Remote procedure call (RPC)

一些开源RPC库

上面介绍了Android和Windows跟随系统提供的RPC框架。此外还有很多独立的RPC框架,例如:

  • Dubbo——阿里巴巴:可以跟Spring框架高度结合;
  • phxrpc——腾讯:跟protobuf高度结合;
  • Motan——新浪微博:跨语言的RPC框架;
  • Dubbox——当当:Dubbo扩展;
  • rpcx:基于 Golang 实现,支持多重通信协议;
  • Navi & Navi-pbrpc
  • Thrift——原Facebook;
  • Avro——hadoop;
  • Finagle——twitter;
  • gRPC——Google:支持多种语言,通信协议是http+protobuf/json;
  • https://github.com/microsoft/IPC:微软基于共享内存实现的Windows IPC C++库;
  • rpclib:基于C++14的RPC库,通信协议默认是TCP/IP,动态绑定是一大亮点,但项目多年未更新了;

详细内容请参考各自主页,本文不做更多展开。

原文链接

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351

推荐阅读更多精彩内容