这是我参与「第五届青训营 」伴学笔记创作活动的第 13 天
1.主要内容
- RPC基本概念
- RPC框架的分层设计和关键指标
- 字节实践
2.本节详细内容
RPC基本概念
RPC就是远程函数调用(Remote Procedure Calls)
RPC调用需要解决的问题:
- 函数映射
- 数据转换成字节流
- 网络传输
RPC带来的问题由框架解决
一次RPC的完整过程
- IDL文件:IDL通过中立方式描述接口,不同平台运行的对象和不同语言编写的程序可以互相通信
- 生成代码:通过编译工具把IDL文件转为语言对应静态库
- 编解码:也叫序列化和反序列化
- 通信协议:规范数据在网络中传输内容和格式,通常包括额外的元数据
- 网络传输:走TCP/UDP传输
RPC分层设计
分层设计
Apache Thrift为例
编解码层
数据格式
- 语言特定格式:例如 java.io.Serializable
- 文本格式:例如 JSON、XML、CSV 等
- 二进制编码:常见有 Thrift 的 BinaryProtocol,Protobuf,实现可以有多种形式,例如 TLV 编码 和 Varint 编码
选型
- 兼容性:需要有良好的可扩展性,支持自动增加新的字段,而不影响老的服务,提高系统的灵活度
- 通用性:技术层面:序列化协议是否支持跨平台、跨语言; 流行程度:序列化和反序列化需要多方参与,流行性度低的协议,跨语言跨平台的包很少也不稳定不成熟
- 性能:空间开销和时间开销来考虑,编码后数据的大小和编码所需要的时长
生成代码和编解码层相互依赖,框架的编解码应当具备扩展任意编解码协议的能力
协议层
是双方确定的交流语义
- 特殊结束符:一个特殊字符作为每个协议单元结束的标示
- 变长协议:以定长加不定长的部分组成,其中定长的部分需要描述不定长的内容长度
协议构造
以Thrift的THeader协议为例
协议解析
网络通信层
Sockets API
套接字编程中客户端必须知道两个信息:服务器的IP地址以及端口号
- Socket函数创建一个套接字,bind将一个套接字绑定到一个地址上。Listen监听进来的连接。Backlog通常情况指定为128即可
- connect客户端向服务器发起连接,accept接受一个连接请求,如果没有连接则会一直阻塞到有连接进来。得到客户端fd后,就可以调用read,write函数和客户端进行通讯,读写方式和其他I/O类似
- read从fd读数据,socket默认是阻塞模式的,如果对方没有写数据,read会一直阻塞
- write从fd写数据,socket默认是阻塞模式的,如果对方没有写数据,write会一直阻塞
- socket关闭套接字,当另一端socket关闭后,尝试读会EOF返回0;尝试写会触发SIGPIPE信号,返回-1和err = EPIPE
网络库
- 提供易用API:封装底层Socket API,连接管理和事件分发
- 功能:协议支持TCP、UDP和UDS等;优雅退出、异常处理等
- 性能:应用层buffer减少copy;高性能定时器、对象池等
- 阻塞 IO 下,耗费一个线程去阻塞在 read(fd) 去等待用足够多的数据可读并返回。
- 非阻塞 IO 下,不停对所有 fds 轮询 read(fd) ,如果读取到 n <= 0 则下一个循环继续轮询。
第一种方式浪费线程(会占用内存和上下文切换开销),第二种方式浪费 CPU 做大量无效工作。而基于 IO 多路复用系统调用实现的 Poll 的意义在于将可读/可写状态通知和实际文件操作分开,并支持多个文件描述符通过一个系统调用监听以提升性能。
网络库的核心功能就是去同时监听大量的文件描述符的状态变化(通过操作系统调用),并对于不同状态变更,高效,安全地进行对应的文件操作。
RPC框架核心指标
稳定性
保障策略
- 熔断:保护调用方,防止被调用的服务出现问题而影响到整个链路
- 限流:保护被调用方,防止大流量把服务干挂
- 超时控制:避免浪费资源在不可用节点上
从某种程度上讲超时、限流和熔断也是一种服务降级的手段
请求成功率
- 负载均衡
- 重试
重试会有放大故障的风险,也就是雪崩,会使下游负载大量升高,甚至直接打挂。防止重试风暴,限制单点重试和限制链路重试
长尾请求
长尾请求一般指明显高于均值的占比较小的请求,业界有常用的P99标准,意思是单个请求响应耗时从小到大排列,顺序处于99%位置的值即为P99值,后面的1%即为长尾请求。
造成长尾请求的原因非常多,常见有网络波动,GC,系统调度等
- Backup Request
先设定一个阈值T3(比超时时间小,通常建议是RPC请求延时的P99),当req发出去超过T3后立即发起重试请求req2,这样就相当于同时有两个请求,只要有一个返回成功就立即结束这次请求
相较于普通超时重试,这种机制可以大大减小整体耗时
易用性
- 开箱即用:合理的默认参数、丰富的文档
- 周边工具:生成代码工具、脚手架工具
扩展性
- Middleware:middleware 会被构造成一个有序调用链逐个执行,比如服务发现、路由、负载均衡、超时控制等
- Option:作为初始化参数
- 核心层是支持扩展的:编解码、协议、网络传输层
- 代码生成工具也支持插件扩展
观测性
- 三件套:Log、Metric 和 Tracing
- 内置观测性服务,用于观察框架内部状态
- 当前环境变量
- 配置参数
- 缓存信息
- 内置 pprof 服务用于排查问题
高性能
- 高性能意味着高吞吐和低延迟,两者都很重要,甚至大部分情况是低延迟重要
- 连接池和多路复用:复用连接,减少频繁建联带来的开销,服务端吞吐可以提升30%
- 高性能编解码协议:Thrift、Protobuf、Flatbuffer 和 Cap'n Proto 等
- 高性能网络库:Netpoll 和 Netty 等
3.企业实践
字节整体架构Kitex
Kitex架构整体概览(概览 | CloudWeGo)
自研网络库
背景
- 原生库无法感知连接状态,存在失效连接,影响连接池复用
- 原生库存在 goroutine 暴涨的风险
Netpoll
[Netpoll](概览 | CloudWeGo)
性能优化
参考 字节跳动 Go RPC 框架 KiteX 性能优化实践
4.课后总结
- 行业中RPC框架各有优劣
- 字节开源的Kitex可以去GitHub上看
- 了解RPC框架的核心指标
- RPC框架的核心三层为重点
5.引用
字节内部课:RPC原理和实现
稀土掘金-后端学习资料
CloudWeGo文档