翻译:REST 和 gRPC 详细比较
译者注:在微服务架构设计,构建API和服务间通信技术选型时,对 REST 和 gRPC 的理解和应用还存在知识盲区,近期看到国外的这篇文章:A detailed comparison of REST and gRPC,将二者进行了详细对比。周末有时间翻译过来,希望能帮到大家!
很长一段时间以来,REST是构建API的唯一“标准”。近年来,出现了新的替代方案。2015年,脸书发布了GraphQL,2016年谷歌紧随其后发布了gRPC,被广泛使用。在本文中,将关注gRPC,并将其与REST进行比较。
概述
下表将概述本文讨论的要点,并显示 REST 和 gRPC 真正的亮点。
| 主题 | REST | gRPC |
|---|---|---|
| 标准化 | 无标准 | 定义明确 |
| 范式 | 以资源为中心 | 以服务方法为中心 |
| 服务模式 | 一元流 | 一元流、客户端流、服务器流和双向流 |
| 要求 | 任何 HTTP、Json解析 | 编程语言需要支持 HTTP/2, gRPC |
| API 设计 | 代码优先 | 设计优先 |
| 默认数据格式 | Json | Protobuf |
| WEB浏览器支持 | 浏览器本地支持 | gRPC WEB |
| 工具 | 成熟的工具 | 特定编程语言工具 |
标准化
REST 缺点之一是缺乏标准化,与其说 REST 是一个API标准,不如说是一种范式。许多人在谈论它时都有不同的含义。对大多数人来说,术语“REST API”是基于HTTP的JSON API;对于其他规范,REST 可以与某些规范(如:HATEOAS或 JSON:API)互换使用。
REST这个术语甚至与HTTP无关。在使用 REST API 时,可能会导致很多混乱。例如,即使没有明确定义,使用者可能会期望某些 REST API 端点的幂等性或可缓存性。
相比之下,gRPC定义得很好。例如,基于 HTTP/2 的gRPC实现非常详细。
基本差异
REST 和 gRPC 的范式并不相同。
REST 一切都以资源为中心,这些资源可以被检索和操作。如果以图书作为示例资源,REST API通常会提供以下端点:
GET /books:检索所有书,可带有筛选和分页结果的参数GET /books/{id}:检索指定ID书POST /books:创建书DELETE /books/{id}:删除书
大多数基于 HTTP 的 REST API 都遵循这种模式。这很好,但某些情况很难表示为 REST API。例如,如果想创建多本书,而不想为每本书重复调用 POST/books(出于性能、幂等性或其他原因),该怎么办?是否应该创建 POST/books/batch 端点?这还是“RESTful”吗?虽然在技术上容易解决,但开发人员之间经常会进行长时间的讨论。
另一方面,gRPC是一个RPC框架,以服务方法为中心。如果以图书 API为例,使用gRPC,将使用以下方法创建 BookService :
GetBooks()GetBook()CreateBook()DeleteBook()
可以随心所欲地命名这些方法,并设置任何需要的参数。如果现在想添加一个方法来创建多本书,没有什么能阻止我们添加 CreateBooks() 方法。gRPC 在设计 API 时提供了更多的“自由”,因为有更少的(自我强加的)限制。
服务模式
gRPC 支持四种服务方法:
- 一元流:发送单个请求,接收单个响应。
- 服务器流:发送单一请求,接收多个响应。
- 客户端流:发送多个请求,接收单一响应。
- 双向流:发送多个请求,接收多个响应。
与只支持一元请求的 REST 相比,gRPC 有一个非常好的优势。在 REST API 中支持其他服务模式需要使用不同的协议,例如:服务器发送的事件或websocket,这不是很“RESTful”。
要求
REST API通常“只适用于”任何类型的 HTTP 版本。只要一种编程语言有一个 HTTP 客户端和一个用于 JSON 解析的库,那么使用 REST API 就轻而易举了。
gRPC 明确需要 HTTP/2 支持,否则它将无法工作。近年来,由于大多数框架都增加了对 HTTP/2 的支持,这已经不再是一个问题了。
由于 gRPC 需要生成代码(用于创建客户端或服务器存根),因此仅部分编程语言支持,查看编程语言列表。
API 设计
REST API 通常是其实现的结果,称为“代码优先(code-first)”。虽然可以先用 OpenAPI 设计API,然后生成服务器存根,但这不是许多开发人员采用的方法。如果有一个 OpenAPI 定义,那么 OpenAPI 定义很可能是从 API 实现生成的。因此,API定义与实现紧密耦合。模型类的更改可能会导致 API 更改,无意中破坏接口规范。
gRPC 使用不同的方法,在实现API之前必须定义API(称为“设计优先(design-first)”)。然后根据 API 定义生成客户端和服务器存根。这需要提前考虑,因为不能直接实现API。
两种方法各有利弊。通常 REST API 方法允许快速迭代;使用gRPC,在调整 API 实现之前首先更改 API 定义,这可能会很烦人。然而,通过显式定义API更加安全,API变动没那么随意。
数据格式
REST 和 gRPC 都可以使用不同的格式来传输数据。大多数 REST API 使用JSON,而 gRPC 默认使用 Protocol Buffers,一种轻便高效的结构化数据存储格式,因此我们将比较这两者。
JSON 对数据类型的支持有限,例如,大数字需要表示为字符串。JOSON 是一种文本格式,便于阅读。字段名序列化会占用一些空间。在一些编程语言中,还需要使用反射来反序列化 JSON 消息,速度慢。
如上所述,gRPC API 和相应的消息类型首先被定义为 Protocol Buffers 。每条消息都是强类型的,对于支持的编程语言,可以自动生成代码以(反)序列化消息。由于它是一种二进制格式,并且不序列化字段名,因此它使用的空间比等效的JSON消息少。但是存在缺点:即序列化消息不可读,并且需要 Protobuf 定义来反序列化消息,可能会影响开发人员的体验。
下面的 JSON 示例将占用大约66个字节(去掉空白)。
{
"persons": [
{
"name": "Max",
"age": 23
},
{
"name": "Mike",
"age": 52
}
]
}
等效的序列化 protobuf 消息将仅使用19个字节。
0x0A070A034D617810170A080A0448616E731034
大消息
Protobuf 设计用于序列化和反序列化内存中的消息。因此,不建议使用 Protobuf/gRPC 传输巨大的消息。大多数 gRPC 实现对单个消息的默认限制为4MB。
使用 REST API 处理大数据量(例如文件上传)是相当直接的。接收到的文件可以被视为流,只需使用很少的内存。这在 gRPC 中需要更多的操作:文件必须在发送方分为几个部分。然后,每个块将作为单独的消息通过客户端流传输方法发送到服务器。服务器接收每个块,并构造数据流,从而产生与 REST API 类似的行为。
浏览器兼容性
浏览器兼容是 REST 真正的闪光点。它由 WEB 浏览器本地支持,因此可以轻松地使用 WEB应用程序中的 REST API。
gRPC 不直接由浏览器支持,因为它需要明确的 HTTP/2 支持和访问某些 HTTP/2 功能,而 WEB浏览器不提供这些功能。作为一种变通方法,可以使用 gRPC Web 协议,使其可供 WEB 浏览器使用。对于某些编程语言,框架中已经包含了gRPC Web 支持。对于其他场景,需要一个代理来将 gRPC 流转换为 gRPC Web 流,反之亦然。与不需要依赖关系的 REST API 相比,gRPC API 在 WEB 上使用更麻烦。
解决方法可以是使用 JSON 转码,这允许开发人员将 gRPC API 公开为 REST API。
工具
gRPC 和 REST 工具在编程语言和框架之间差异很大。在某些情况下,gRPC 感觉更“原生”,而在另一些情况下,REST 工具则更高级。
对 gRPC 的编程语言支持更为重要,因为需要工具来生成特定编程语言的客户端和服务器存根。对于不受支持的编程语言,不建议使用 gRPC 。
由于 REST API 存在的时间要长得多,因此存在更多有助于构建、测试和部署 REST API 的工具。它们的功能通常比 gRPC 工具更高级。
小结
对于“应该使用 REST 还是 gRPC ?” 没有明确的答案。有些 API 有独特的用例,gRPC 或 REST 可能更适合这些用例。
REST 和 gRPC 都有各自的优点和缺点。
从 WEB 应用程序中使用 REST API 通常更容易。REST也得到了更广泛的使用,这使得开发人员使用它更简单。
gRPC 在服务器到服务器的通信(例如:微服务之间)方面无疑具有优势。能够共享准确的API定义并用多种编程语言创建 API 客户端,这是一个巨大的优势。
dotNET兄弟会-公众号
专注.Net开源技术及跨平台开发!致力于构建完善的.Net开发技术文库!为.Net爱好者提供学习交流家园!

翻译:REST 和 gRPC 详细比较的更多相关文章
- 如何在golang中打印grpc详细日志
最近捣鼓fabric,在一个tls证书问题上纠结挺久,连接orderer服务时候,grpc日志总是冷冰冰的显示这个信息 Orderer Client Status Code: (2) CONNECTI ...
- 详细教程丨使用Prometheus和Thanos进行高可用K8S监控
本文转自Rancher Labs 介 绍 Prometheus高可用的必要性 在过去的几年里,Kubernetes的采用量增长了数倍.很明显,Kubernetes是容器编排的不二选择.与此同时,Pro ...
- Python批量图片识别并翻译——我用python给女朋友翻译化妆品标签
Python批量图片识别并翻译--我用python给女朋友翻译化妆品标签 最近小编遇到一个生存问题,女朋友让我给她翻译英文化妆品标签.美其名曰:"程序猿每天英语开发,英文一定很好吧,来帮我翻 ...
- 【硬核】MMU是如何完成地址翻译的
目录 1. 什么是虚拟内存? 2. 虚拟内存的作用 3. 虚拟内存与物理内存 3.1 CPU存取数据 3.2 物理地址常用术语 3.3 虚拟地址常用术语 3.4 页表常用术语 3.5 页命中/缺页 4 ...
- [译]基于GPU的体渲染高级技术之raycasting算法
[译]基于GPU的体渲染高级技术之raycasting算法 PS:我决定翻译一下<Advanced Illumination Techniques for GPU-Based Volume Ra ...
- 系统间通信(10)——RPC的基本概念
1.概述 经过了详细的信息格式.网络IO模型的讲解,并且通过JAVA RMI的讲解进行了预热.从这篇文章开始我们将进入这个系列博文的另一个重点知识体系的讲解:RPC.在后续的几篇文章中,我们首先讲解R ...
- 【Other】最近正在看的
待看: https://my.oschina.net/yunfound/blog/141222 https://www.zhihu.com/question/22925358 http://study ...
- 分布式系统间通信之RPC的基本概念(六)
RPC(Remote Procedure Call Protocol)远程过程调用协议.一个通俗的描述是:客户端在不知道调用细节的情况下,调用存在于远程计算机上的某个对象,就像调用本地应用程序中的对象 ...
- Q_D宏
Qt 源码中有很多Q_Q和Q_D宏,使用这些宏的地方总会看到有q指针和d指针,查了查KDE文档,大体搞清了其中的机理,欧也!Qt的这些私有数据访问策略还是挺值得借鉴.下面就简单总结一下. 访问器 , ...
- 『WPF』DataGrid的使用
原文 『WPF』DataGrid的使用 几点说明 这里主要是参考了MSDN中关于DataGrid的说明 这里只会简单说明在WPF中,DataGird最简单的使用方法 对于MSDN中的翻译不会很详细,也 ...
随机推荐
- 为什么HashMap查找比List快很多?
做两数之和这道题目时,引发了一个思考: 为什么两者运行时间相差如此之大???好残忍,我List比你HashMap到底差在哪**** 于是我一顿查资料.... 战犯哈希算法登场 哈希算法会根据你要存入的 ...
- Android和adb命令
一.名词解释 1.SDK:是软件开发工具包 2.activity(活动):驱使软件运行的一段程序,软件系统和用户进行交互的界面叫一个活动 二.adb命令 1.查看连接的设备:adb devices 2 ...
- 使用Electron-packager打包已有的web项目,发布客户端
1.先拉electron代码 git clone https://github.com/electron/electron-quick-start 2.将web项目拷贝到electron-quick- ...
- Java中使用List的add方法后元素相同问题
在写JavaWeb时,我在后端通过JDBC读取了数据后逐个使用List.add()方法添加元素并通过request方法传给jsp页面解析,但是添加以后出现了在列表里有n个(假设添加了n个元素)最后一个 ...
- Defi开发简介
Defi开发简介 介绍 Defi是去中心化金融的缩写, 是一项旨在利用区块链技术和智能合约创建更加开放,可访问和透明的金融体系的运动. 这与传统金融形成鲜明对比,传统金融通常由少数大型银行和金融机构控 ...
- window远程桌面
此文档概述如何开启win8.win8.1系统的远程桌面连接服务,可以让我们从一台电脑远程连接的其他电脑! 同时按"win键+R键",再打开的运行对话框中输入"contro ...
- Vue路由实现的底层原理
在Vue中利用数据劫持defineProperty在原型prototype上初始化了一些getter,分别是router代表当前Router的实例 . router代表当前Router的实例.rout ...
- Java学习笔记02
1. 运算符和表达式 运算符 就是对常量或者变量进行操作的符号. 如:+ - * / 表达式 用运算符把常量或者变量连接起来的,符合Java语法的式子就是表达式. 如:a + b ...
- Python 将函数存储在模块中
将函数存储在模块中 将函数存储在被称为模块的独立文件中,在将模块导入到主程序中 import语句允许在当前运行的程序文件中使用模块中的代码 通过将函数存储在独立的文件中,可影藏程序的代码细节,将重点放 ...
- B/S结构系统的会话机制(session)
B/S结构系统的会话机制(session) 目录 B/S结构系统的会话机制(session) 每博一文案 1. session 会话机制的概述 2. 什么是 session 的会话 3. sessio ...