微服务实战(二):使用API Gateway
【编者的话】本系列的第一篇介绍了微服务架构模式。它讨论了采用微服务的优点和缺点,除了一些复杂的微服务,这种模式还是复杂应用的理想选择。
当你决定将应用作为一组微服务时,需要决定应用客户端如何与微服务交互。在单体式程序中,通常只有一组冗余的或者负载均衡的服务提供点。在微服务架构中,每一个微服务暴露一组细粒度的服务提供点。在本篇文章中,我们来看它如何影响客户端到服务端通信,同时提出一种API Gateway
的方法。
介绍
假定你正在为在线购物应用开发一个原生手机客户端。你需要实现一个产品最终页来展示商品信息。
例如,下面的图展示了你在亚马逊Android
客户端上滑动产品最终页时看到的信息。
虽然这是一个智能手机应用,这个产品最终页展示了非常多的信息。例如,不仅这里有产品基本信息(名字、描述和价格),还有以下内容:
- 购物车中的物品数
- 下单历史
- 用户评论
- 低库存警告
- 快递选项
各式各样的推荐,包括经常跟这个物品一起被购买的产品、购买该物品的其他顾客购买的产品以及购买该产品的顾客还浏览了哪些产品。
可选的购物选项
当采用一个单体式应用架构,一个移动客户端将会通过一个REST
请求(GET api.company.com/productdetails/productId
)来获取这些数据。一个负载均衡将请求分发到多个应用实例之一。应用将查询各种数据库并返回请求给客户端。
相对的,若是采用微服务架构,最终页上的数据会分布在不同的微服务上。下面列举了可能与产品最终页数据有关的一些微服务:
- 购物车服务 — 购物车中的物品数
- 下单服务 — 下单历史
- 分类服务 — 基本产品信息,如名字、图片和价格
- 评论服务 — 用户评论
- 库存服务 — 低库存警告
- 快递服务 — 快递选项、截止时间、来自不同快递API的成本计算
- 推荐服务 — 推荐产品
我们需要决定移动客户端如何访问这些服务。请看下面这几种方式
客户端到微服务直接通信
理论上说,一个客户端可以直接给多个微服务中的任何一个发起请求。每一个微服务都会有一个对外服务端(https://serviceName.api.company.name)。这个URL
可能会映射到微服务的负载均衡上,它再转发请求到具体节点上。为了搜索产品细节,移动端需要向上述微服务逐个发请求。
不幸的是,这个方案有很多困难和限制。其中一个问题是客户端的需求量与每个微服务暴露的细粒度API
数量的不匹配。如图中,客户端需要7次单独请求。在更复杂的场景中,可能会需要更多次请求。例如,亚马逊的产品最终页要请求数百个微服务。虽然一个客户端可以通过LAN
发起很多个请求,但是在公网上这样会很没有效率,这个问题在移动互联网上尤为突出。这个方案同时会导致客户端代码非常复杂。
另一个存在的问题是客户端直接请求微服务的协议可能并不是web
友好型。一个服务可能是用Thrift
的RPC
协议,而另一个服务可能是用AMQP
消息协议。它们都不是浏览或防火墙友好的,并且最好是内部使用。应用应该在防火墙外采用类似HTTP
或者WEBSocket
协议。
这个方案的另一个缺点是它很难重构微服务。随着时间的推移,我们可能需要改变系统微服务目前的切分方案。例如,我们可能需要将两个服务合并或者将一个服务拆分为多个。但是,如果客户端直接与微服务交互,那么这种重构就很难实施。
由于上述三种问题的原因,客户端直接与服务器端通信的方式很少在实际中使用。
采用一个API Gateway
通常来说,一个更好的解决办法是采用API
Gateway
的方式。API
Gateway
是一个服务器,也可以说是进入系统的唯一节点。这跟面向对象设计模式中的Facade
模式很像。API
Gateway
封装内部系统的架构,并且提供API
给各个客户端。它还可能有其他功能,如授权、监控、负载均衡、缓存、请求分片和管理、静态响应处理等。下图展示了一个适应当前架构的API
Gateway
。
API
Gateway
负责请求转发、合成和协议转换。所有来自客户端的请求都要先经过API
Gateway
,然后路由这些请求到对应的微服务。API
Gateway
将经常通过调用多个微服务来处理一个请求以及聚合多个服务的结果。它可以在web
协议与内部使用的非Web友好型协议间进行转换,如HTTP
协议、WebSocket
协议。
API
Gateway
可以提供给客户端一个定制化的API
。它暴露一个粗粒度API
给移动客户端。以产品最终页这个使用场景为例。API Gateway
提供一个服务提供点(/productdetails?productid=xxx
)使得移动客户端可以在一个请求中检索到产品最终页的全部数据。API Gateway
通过调用多个服务来处理这一个请求并返回结果,涉及产品信息、推荐、评论等。
一个很好的API Gateway
例子是Netfix API Gateway。Netflix
流服务提供数百个不同的微服务,包括电视、机顶盒、智能手机、游戏系统、平板电脑等。起初,Netflix
视图提供一个适用全场景的API
。但是,他们发现这种形式不好用,因为涉及到各式各样的设备以及它们独特的需求。现在,他们采用一个API Gateway
来提供容错性高的API,针对不同类型设备有相应代码。事实上,一个适配器处理一个请求平均要调用6到8个后端服务。Netflix API Gateway
每天处理数十亿的请求。
API Gateway的优点和缺点
如你所料,采用API Gateway
也是优缺点并存的。API Gateway
的一个最大好处是封装应用内部结构。相比起来调用指定的服务,客户端直接跟gatway
交互更简单点。API Gateway
提供给每一个客户端一个特定API
,这样减少了客户端与服务器端的通信次数,也简化了客户端代码。
API Gateway
也有一些缺点。它是一个高可用的组件,必须要开发、部署和管理。还有一个问题,它可能成为开发的一个瓶颈。开发者必须更新API Gateway
来提供新服务提供点来支持新暴露的微服务。更新API Gateway
时必须越轻量级越好。否则,开发者将因为更新Gateway
而排队列。但是,除了这些缺点,对于大部分的应用,采用API Gateway
的方式都是有效的。
实现一个API Gateway
既然我们已经知道了采用API Gateway
的动机和优缺点,下面来看在设计它时需要考虑哪些事情。
性能和可扩展性
只有少数公司需要处理像Netflix
那样的规模,每天需要处理数十亿的请求。但是,对于大多数应用,API Gateway
的性能和可扩展性也是非常重要的。因此,创建一个支持同步、非阻塞I/O
的API Gateway
是有意义的。已经有不同的技术可以用来实现一个可扩展的API Gateway
。在JVM上,采用基于NIO
技术的框架,如Netty
,Vertx
,Spring Reactor
或者JBoss Undertow
。Node.js
是一个非JVM
的流行平台,它是一个在Chrome
的JavaScript
引擎基础上建立的平台。一个可选的方案是NGINX Plus。NGINX Plus
提供一个成熟的、可扩展的、高性能web
服务器和反向代理,它们均容易部署、配置和二次开发。NGINX Plus
可以管理授权、权限控制、负载均衡、缓存并提供应用健康检查和监控。
采用反应性编程模型
对于有些请求,API Gateway
可以通过直接路由请求到对应的后端服务上的方式来处理。对于另外一些请求,它需要调用多个后端服务并合并结果来处理。对于一些请求,例如产品最终页面请求,发给后端服务的请求是相互独立的。为了最小化响应时间,API Gateway
应该并发的处理相互独立的请求。但是,有时候请求之间是有依赖的。API Gateway
可能需要先通过授权服务来验证请求,然后在路由到后端服务。类似的,为了获得客户的产品愿望清单,需要先获取该用户的资料,然后返回清单上产品的信息。这样的一个API 组件是Netflix Video Grid。
利用传统的同步回调方法来实现API
合并的代码会使得你进入回调函数的噩梦中。这种代码将非常难度且难以维护。一个优雅的解决方案是采用反应性编程模式来实现。类似的反应抽象实现有Scala
的Future,Java8
的CompletableFuture和JavaScript
的Promise。基于微软.Net
平台的有Reactive Extensions(Rx)。Netflix
为JVM
环境创建了RxJava
来使用他们的API Gateway
。同样地,JavaScript
平台有RxJS
,可以在浏览器和Node.js
平台上运行。采用反应编程方法可以帮助快速实现一个高效的API Gateway
代码。
服务调用
一个基于微服务的应用是一个分布式系统,并且必须采用线程间通信的机制。有两种线程间通信的方法。一种是采用异步机制,基于消息的方法。这类的实现方法有JMS
和AMQP
。另外的,例如Zeromq
属于服务间直接通信。还有一种线程间通信采用同步机制,例如Thrift
和HTTP
。事实上一个系统会同时采用同步和异步两种机制。由于它的实现方式有很多种,因此API Gateway
就需要支持多种通信方式。
服务发现
API Gateway
需要知道每一个微服务的IP
和端口。在传统应用中,你可能会硬编码这些地址,但是在现在云基础的微服务应用中,这将是个简单的问题。基础服务通常会采用静态地址,可以采用操作系统环境变量来指定。但是,探测应用服务的地址就没那么容易了。应用服务通常动态分配地址和端口。同样的,由于扩展或者升级,服务的实例也会动态的改变。因此,API Gateway需要采用系统的服务发现机制,要么采用服务端发现,要么是客户端发现。后续的一篇文章将会更详细的介绍这部分。如果采用客户端发现服务,API Gateway
必须要去查询服务注册处,也就是微服务实例地址的数据库。
处理部分失败
在实现API Gateway
过程中,另外一个需要考虑的问题就是部分失败。这个问题发生在分布式系统中当一个服务调用另外一个服务超时或者不可用的情况。API Gateway
不应该被阻断并处于无限期等待下游服务的状态。但是,如何处理这种失败依赖于特定的场景和具体服务。例如,如果是在产品详情页的推荐服务模块无响应,那么API Gateway
应该返回剩下的其他信息给用户,因为这些信息也是有用的。推荐部分可以返回空,也可以返回固定的顶部10个给用户。但是,如果是产品信息服务无响应,那么API Gateway
就应该给客户端返回一个错误。
在缓存有效的时候,API Gateway
应该能够返回缓存。例如,由于产品价格变化并不频繁,API Gateway
在价格服务不可用时应该返回缓存中的数值。这类数据可以由API Gateway
自身来缓存,也可以由Redis
或Memcached
这类外部缓存实现。通过返回缓存数据或者默认数据,API Gateway
来确保系统错误不影响到用户体验。
Netflix Hystrix对于实现远程服务调用代码来说是一个非常好用的库。Hystrix
记录那些超过预设定的极限值的调用。它实现了circuit break
模式,使得可以将客户端从无响应服务的无尽等待中停止。如果一个服务的错误率超过预设值,Hystrix
将中断服务,并且在一段时间内所有请求立刻失效。Hystrix
可以为请求失败定义一个fallback
操作,例如读取缓存或者返回默认值。如果你在用JVM
,就应该考虑使用Hystrix
。如果你采用的非JVM
环境,那么应该考虑采用类似功能的库。
总结
对于大多数微服务基础的应用,实现一个API Gateway
都是有意义的,它就像是进入系统的一个服务提供点。API Gateway
负责请求转发、请求合成和协议转换。它提供给应用客户端一个自定义的API
。API Gateway
可以通过返回缓存或者默认值的方式来掩盖后端服务的错误。在本系列的下一篇文章中,我们将讨论服务间的通信问题。
微服务实战(二):使用API Gateway相关推荐
- SpringCloud Alibaba微服务实战(七) - 路由网关(Gateway)全局过滤
说在前面 全局过滤器作用于所有的路由,不需要单独配置,我们可以用它来实现很多统一化处理的业务需求,比如权限认证,IP 访问限制,监控,限流等等. 创建路由网关(Gateway)启动服务cloud-ac ...
- SpringCloud Alibaba微服务实战(六) - 路由网关(Gateway)
什么是 Spring Cloud Gateway? Spring Cloud Gateway 是 Spring 官方基于 Spring 5.0,Spring Boot 2.0 和 Project Re ...
- SpringCloud Alibaba微服务实战(二) - Nacos服务注册与restTemplate消费
说在前面 基础环境搭建,理论,请看上一篇,在这就不扯理论了,直接上代码. 项目结构 代码实现 第一步:在父pom的项目中引入dependencyManagement 在引入父pom之前咱们先来回顾下d ...
- SpringCloud Alibaba微服务实战(五) - Sentinel实现限流熔断
什么是Sentinel? 请查看文章:SpringCloud Alibaba微服务实战(一) - 基础环境搭建 构建服务消费者cloud-sentinel进行服务调用 服务创建请查看文章:Spring ...
- SpringCloud Alibaba微服务实战(三) - Nacos服务创建消费者(Feign)
什么是Feign Feign 是一个声明式的伪 Http 客户端,它使得写 Http 客户端变得更简单.使用 Feign,只需要创建一个接口并注解.它具有可插拔的注解特性,可使用 Feign 注解和 ...
- controller调用controller的方法_SpringCloud Alibaba微服务实战三 - 服务调用
导读:通过前面两篇文章我们准备好了微服务的基础环境并让accout-service 和 product-service对外提供了增删改查的能力,本篇我们的内容是让order-service作为消费者远 ...
- java 限流熔断_SpringCloud Alibaba微服务实战五 - 限流熔断
简介 Sentinel是面向分布式服务框架的轻量级流量控制框架,主要以流量为切入点,从流量控制,熔断降级,系统负载保护等多个维度来维护系统的稳定性.在SpringCloud体系中,sentinel主要 ...
- 手把手,嘴对嘴教你Spring Cloud 微服务实战 -- 前言
Spring Cloud 总结 博主接触到Spring Cloud 大概已经一年多了,当时Spring Cloud微服务框架已经是潮流了,不会一点都不好意思出去面试.并且主流技术基本上都在谈论微服务, ...
- 微服务实战(六):选择微服务部署策略
http://dockone.io/article/1066 微服务实战(六):选择微服务部署策略 [编者的话]这篇博客是用微服务建应用的第六篇,第一篇介绍了微服务架构模板,并且讨论了使用微服务的优缺 ...
- 微服务实战(七):从单体式架构迁移到微服务架构
http://dockone.io/article/1266 希望读者通过本系列文章对微服务优缺点有一个比较好的理解,以及何时使用这种架构.也许微服务架构比较适合你的应用.也许你正在开发一个大型.复杂 ...
最新文章
- AP计算机科学有几门,2017新增AP计算机科学考试科目
- Grafana中多租户设置
- BZOJ 1188: [HNOI2007]分裂游戏(multi-nim)
- 曾经迷茫的起点,多年后终于明白了(续)
- 雨林木风win11 64位安全旗舰版镜像V2021.09
- oracle设置打印机,使用 Oracle Solaris 打印管理器设置网络连接的打印机
- 世界500强的科技从业者,依旧逃不出买房难的宿命
- 计算机网络-交换机配置
- 修复easyMule for Mac 2.0崩溃造成的任务丢失
- 用PHPphpstudy写一个可以登录的简单网页
- Linux分卷压缩zip文件命令,linux下的几种分卷压缩命令
- 大一新生必看,自学必看,里昂详解数据结构之堆栈
- 机器学习第三章笔记——决策树
- 括弧匹配检验(括号匹配问题)
- 树莓派与DS18B20温度传感器模块的使用
- 大众汽车平台PQ、MBQ简介
- 中国农超对接模式产业竞争动态格局与建设经营分析报告2022-2028年
- 100道Java实习生需要掌握的面试问题
- TED演讲字幕下载脚本的使用方法
- android 儿童 游戏,7-10岁儿童游戏大全