jjzjj

技术选型:如何选择REST、GraphQL 和 gRPC

迷路的架构师 2023-03-28 原文
REST、GraphQL 和 gRPC 是现代 Web 应用程序中最流行的 3 种 API 开发技术。那么在做技术选型时,三者要如何选择呢?

在本文中,我们将一起对比 REST、GraphQL 和 gRPC 的特性和用法。

REST——最流行的技术

REST

Representational State Transfer (REST) 是现代 Web 开发中最流行的 API 开发技术。它是一个无状态的数据传输架构。客户端请求时会包含该请求所需的所有详细信息,但是服务器不保留客户端的状态。

REST API 支持 HTTP 原生缓存 header 并使用 HTTP 方法(POST、GET、PUT、PATCH 和 DELETE)来操作数据。因为 REST 的学习门槛较低,所以大家都能轻松使用 REST。

REST 易于扩展且可靠,如果我们还在犹豫不决时,可以优先选择它。

REST的好处

  • 可以放心地使用标准 HTTP 方法实现 CRUD 操作。
  • REST 已经很成熟,有完善的文档,上手简单。
  • 支持缓存。
  • 友好的可扩展性,并提供客户端和服务器之间的分离。
  • 可以轻松地将其集成到应用程序中。

REST的缺点

  • 存在过度获取和获取不足的问题。当API返回比实际需要的数据更多时,就会发生过度获取。这可能会导致不必要的网络流量、较慢的性能和额外的带宽使用。获取不足发生在API没有返回特定用例所需的所有必要数据时,需要多个请求才能检索所有所需信息。这也可能导致较慢的性能和增加的网络流量,以及更复杂的代码库。
  • 不能维持状态。
  • 相对来说比较大的 payload。
  • 随着应用程序的扩展,端点的数量急剧增加。
  • 不容易更新数据库模式或数据结构。

何时选择 REST

如果没有特定要求,REST 是最佳选择。如果是开发新手,那么使用 REST 是完美的选择,因为它的学习曲线较浅。此外,它还拥有庞大的生态系统,你可以轻松找到问题的解决方案。

在处理较大的请求量和带宽有限时最好使用 REST,因为可以使用它的缓存支持来提高性能。

总的来说,如果你的应用程序没有明确需要使用 GraphQL 或 gRPC,那么就可以使用 REST。

GraphQL——客户端驱动的标准

GraphQL 是 2015 年推出的一种数据查询语言,支持开发人员精确定位和获取他们需要的数据。与 REST 相比,GraphQL 是一种客户端驱动的方法,客户端可以决定需要什么数据、如何获取数据和格式。它还解决了过度获取和获取不足的问题,因为客户端可以明确指定所需的数据。

GraphQL 使用查询、变更和订阅来操作数据。

  • 查询:从服务器请求数据。
  • 变更:修改服务器端数据。
  • 订阅:在数据更新时,通过订阅获得实时更新的数据。
GitHub 是使用 GraphQL 的最大公司之一。它在 2016 年从 REST 转向 GraphQL,极大地帮助了 GitHub 的快速增长。

GraphQL 的好处

  • 非常灵活,可以准确地满足客户的需求。
  • 没有过度获取和获取不足的问题。
  • 主流语言支持,包括 JavaScript、Java、Python、Ruby 和 PHP。
  • 允许自定义数据的结构。
  • 单个查询可以包含来自多个资源的字段。

GraphQL 的缺点

  • 查询可能很复杂。
  • 缺乏内置的缓存支持。
  • 与 REST 相比,学习 GraphQL 更具挑战性。
  • 默认不支持文件上传。

何时选择 GraphQL

当查询包含数据库的许多记录时,GraphQL 是最佳选择。你可以使用 GraphQL 消除过度获取,并仅查询必要数据以提高应用程序性能。此外,GraphQL 非常适合需要从多个资源聚合数据的情况。

当你还不完全了解客户端使用 API 的原理时,也可以使用 GraphQL。使用 GraphQL 时,你无需预先定义严格的协议,可以根据客户反馈逐步构建 API。

gRPC——一种以性能为导向的技术

gRPC 是 Google 于 2016 年推出的远程过程调用的进化版本。它是一种轻量级解决方案,使用最少的资源提供最大的性能。

gRPC 遵循基于协议的通信方法。它要求客户端和服务器在开始通信之前都有协议。gRPC 使用 Protobuf(一种声明性语言)创建协议,并使用选定的语言为客户端和服务器生成兼容的代码。

gRPC支持的通信方式有4种:

  • Unary :客户端发送一个请求并等待单个响应。
  • Server streaming :客户端发送一个请求并接收多个响应。
  • Client streaming :客户端发送多个请求并等待单个响应。
  • Bidirectional streaming :客户端发送多个请求并接收多个响应。

gRPC 的好处

  • 开源。开发人员可以根据需要对其进行修改。
  • 支持多种语言,包括 JavaScript、Java、C、C++、C#、Kotlin、Python、Go 和 PHP。
  • 能够进行负载均衡。
  • 与 REST API 相比,它默认使用 HTTP2 来减少延迟。
  • 使用二进制格式序列化数据。
  • 支持全双工流媒体。

gRPC 的缺点

  • 学习曲线较陡峭:与传统的REST API相比,gRPC需要掌握新的概念和技术,例如Protocol Buffers和流。
  • 可读性差:由于使用二进制编码,gRPC的消息不像JSON或XML那样易于人类阅读和理解。
  • 难以调试:由于消息是二进制编码的,调试gRPC服务可能比调试REST API更加困难。
  • 不适合小型应用:对于只有少量服务和少量数据的小型应用程序来说,gRPC可能过于复杂,增加了不必要的开销。
  • 不支持Web浏览器:由于gRPC使用HTTP/2协议,而Web浏览器目前还不支持HTTP/2协议的所有功能,因此不能在Web浏览器中使用gRPC。

何时选择 gRPC

  1. 需要高效的数据传输:由于gRPC使用二进制协议,因此比JSON和XML等文本协议更快、更轻量级。
  2. 需要高可靠性:gRPC的基于HTTP/2协议的传输层提供了许多功能,例如流控制、连接复用和头部压缩等,这些功能可以提高可靠性和性能。
  3. 需要高效的多语言通信:gRPC支持多种编程语言,并提供了自动生成代码的工具,因此不需要手动编写跨语言的代码。
  4. 需要支持多种请求和响应类型:gRPC支持四种类型的通信方式(Unary、Server streaming、Client streaming和Bidirectional streaming),因此可以选择最适合特定用例的通信方式。
  5. 需要更好的API管理:gRPC提供了强大的API管理工具,例如gRPC-Gateway和Envoy等,这些工具可以提高API的可发现性、文档化和测试。
gRPC 可以用在微服务架构中来处理服务之间的通信,因为它可以与用不同语言编写的服务进行通信。

结论

选择REST、GraphQL和gRPC取决于你的具体场景和需求,基本原则总结如下:

  1. REST:REST适合简单的API和Web服务,例如传统的CRUD操作。它通常更易于理解和使用,并且具有广泛的支持和工具生态系统。
  2. GraphQL:GraphQL适合需要灵活性和高级查询功能的应用程序。如果你的应用程序需要从多个资源聚合数据,或者需要更好地控制数据的格式和粒度,则GraphQL是一个不错的选择。
  3. gRPC:gRPC适合需要高效和可靠数据传输的应用程序。如果你需要在多种编程语言之间进行高效通信,并且希望提供更高的性能和可靠性,则gRPC是一个不错的选择。
不过,REST、GraphQL和gRPC并不是相互排斥的选择。在实际情况下,你可以结合使用,以满足具体需求和场景。


有关技术选型:如何选择REST、GraphQL 和 gRPC的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  4. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  5. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  7. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  8. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  9. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

    我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

  10. ruby - 如何使用文字标量样式在 YAML 中转储字符串? - 2

    我有一大串格式化数据(例如JSON),我想使用Psychinruby​​同时保留格式转储到YAML。基本上,我希望JSON使用literalstyle出现在YAML中:---json:|{"page":1,"results":["item","another"],"total_pages":0}但是,当我使用YAML.dump时,它不使用文字样式。我得到这样的东西:---json:!"{\n\"page\":1,\n\"results\":[\n\"item\",\"another\"\n],\n\"total_pages\":0\n}\n"我如何告诉Psych以想要的样式转储标量?解

随机推荐