软件架构场景之—— BFF:如何处理好微服务之间千丝万缕的关系?_bff:如何处理好微服务之间千丝万缕的关系?-程序员宅基地

技术标签: 微服务多关联调用  微服务  bff  

业务场景

之前设计的一个供应链系统中,它包含了商品、销售订单、加盟商、门店运营、门店工单等服务,涉及了各种用户角色,比如总部商品管理、总部门店管理、加盟商员工、门店人员等,而且每个部门的角色还会进行细分。而且这个系统中还包含了两个客户端 App:一个面向客户,另一个面向公司员工和加盟商

此时,整个供应链系统的架构如下图所示

上图中的网关层主要负责路由、认证、监控、限流熔断等工作

  • 路由: 所有的请求都需要通过网关层进行处理,网关层再根据 URI 将请求指向对应的后台服务,如果同一个服务存在多个服务器节点,网关层还将承担负载均衡的工作
  • 认证: 对所有的请求进行集中认证鉴权
  • 监控: 记录所有的 API 请求数据,API 管理系统能对 API 调用实现管理和性能监控
  • 限流熔断: 流量过大时,可以在网关层实现限流。如果后台服务响应延时或故障,可以主动在调用端的上游服务做熔断,以此保护后端服务资源,同时不影响用户体验

此时,架构看起来是不是挺完美?且市面上标准的 Spring Cloud 架构都是这样做的。不过,这个架构会出现一些问题,下面先通过几个例子来看看

 

案例一:

在这个供应链系统中,很多界面都需要显示多个服务数据,比如在一个 App 首页中,针对门店运营人员,需要显示工单数量、最近的工单、销售订单数据、最近待处理的订单、低于库存安全值的商品等信息

此时第一个问题来了,在接口设计过程中,经常纠结将两个客户端 App 调用的接口存放在哪个服务中?以至于决策效率低下,而且还会出现职责划分不统一的情况

最终决定将第一个接口存放在门店服务中,此时调用关系如下图所示

并将第二个接口存放在工单服务中,此时调用关系如下图所示

 

案例二:

一个用户的提交操作常常需要修改多个服务数据,比如一个提交工单的操作,我们需要修改库存、销售订单状态、工单等数据

此时第二个问题出现了,因为这样的需求非常多,所以服务经常被其他多个服务调来调去,导致服务之间的依赖非常混乱,最终服务调用关系如下图所示

通过上图,发现服务间的依赖问题给技术迭代带来了地狱般的体验,为了解决这 2 个问题,最终决定抽象一个 API 层

 

API 层

一般来说,客户端的接口需要满足聚合、分布式调用、装饰这三种需求

  • 聚合:一个接口需要聚合多个后台服务返回的数据,并将数据返回给客户端
  • 分布式调用:一个接口可能需要依次调用多个后台服务,才能实现多个后台服务的数据修改
  • 装饰:一个接口需要重新装饰后台返回的数据,比如删除一些字段或者对某些字段进行封装,然后组成客户端需要的数据

因此,决定在客户端与后台服务之间增加一个新的 API 层,专门用来满足上面的三点需求,此时整个架构如下图所示

从图中我们发现,所有请求经过网关后,全部交由一个共用的 API 层进行处理,而该 API 层没有自己的数据库,它的主要职责是调用其他后台服务

通过这样的设计方案后,以上两个问题就得到了很多地解决

  • 应该将某个接口放在哪个服务的纠结次数减少了: 如果是聚合、装饰、分布式的调用逻辑,直接把它们放在 API 层。如果是要落库或者查询数据库的逻辑,目标数据在哪个服务中,我们就把数据和逻辑放在哪个服务中
  • 后台服务之间的依赖也大幅减少了: 目前的依赖关系只有 API 层调用各个后台服务

 

客户端适配问题

在这个供应链系统中,一系列的接口主要供各种客户端(比如 App、H5、PC 网页、小程序等)进行调用,此时的调用关系如下图所示

不过,这种设计方案会存在 3 个问题

  • 不同客户端的页面细节的需求可能不一样,比如 App 的功能比重大,就会要求页面中多放一些信息,而小程序的功能比重小,同样的页面就会要求少放一些信息,以至于后台服务中同一个 API 需要针对不同客户端实现不同适配;
  • 客户端经常需要进行一些轻微的改动,比如增加一个字段/删除一个字段,此时必须采取数据最小化原则来缩减客户端接口的响应速度。而且,为了客户端这种细微而频繁的改动,后台服务经常需要同步发版;
  • 结合 #1 和 #2 发现,在后台服务的发版过程中,常常需要综合考虑不同客户端的兼容问题,这无形中增加了 API 层为不同客户端做兼容的复杂度

这时该如何解决呢?我们就可以考虑使用 BFF 了

 

BFF(Backend for Front)

BFF 不是一个架构,而是一个设计模式,它的主要职责是为前端设计出优雅的后台服务,即一个 API。一般而言,每个客户端都有自己的 API 服务,此时整个架构如下图所示

从上图可以看到:不同的客户端请求经过同一个网关后,它们都将分别重定向到为对应客户端设计的 API 服务中。因为每个 API 服务只能针对一种客户端,所以它们可以对特定的客户端进行专门优化。而去除了兼容逻辑的 API 显得更轻便,响应速度还比通用的 API 服务更快(因为它不需要判断不同客户端的逻辑)

除此之外,每种客户端还可以实现自己发布,不需要再跟着其他客户端一起排期

此时的方案挺完美了吧?还不完美,因为上面的方案属于一个通用架构。在实际业务中,还需要结合实际业务来定,下面深入说明一下实际业务需求

前面列出了 5 种服务,实际上,整个供应链系统将近有 100 种服务。因为它是一个非常庞大的系统,且整个业务链条的所有工作都包含在这个系统中,比如新零售、供应链、财务、加盟商、售后、客服等,这就需要几百号研发人员同时进行维护

因为共同维护一个 App、PC 界面、新零售、售后、加盟商,还有各自的小程序和 H5,所以为了实现业务解耦和分开排期,每个部门需要各自维护自己的 API 服务,而且 App 与 PC 前端也需要根据部门实现组件化,此时的架构如下图所示

 

技术架构上如何实现?

整套架构还是基于 Spring Cloud 设计的,如下图所示

下面简单介绍下图中网关、API服务、后台服务的作用

  • 网关: 网关使用的是 Spring Cloud Zuul,Zuul 将拉取的注册存放在 ZooKeeper 的 API 服务中,然后通过 Feign 调用 API 服务
  • API 服务: API 服务其实就是一个 Spring Web 服务,它没有自己的数据库,主要职责是聚合、分布式调用及装饰数据,并通过 Feign 调用后台服务
  • 后台服务: 后台服务其实也是一个 Spring Web 服务,它有自己的数据库和缓存

此时的方案看着很完美了,不过它会出现 API 之间代码重复问题。此时我们该如何解决?

 

如何解决 API 之间代码重复问题?

虽然 H5 与小程序的布局不同,但是页面中很多功能一致,也就是说重复的代码逻辑主要存在 PC API 和 App API 中

然而,针对重复代码的问题,不同部门在设计时会呈现 3 种不同的逻辑

  • 某些部门将这些重复的代码存放在一个 JAR 中,让几个 API 服务实现共用;
  • 某些部门将这些重复的代码抽取出来,然后存放在一个叫 CommonAPI 的独立 API 服务中,其他 API 服务直接调用这个 Common API 就行;
  • 某些部门因为重复逻辑少,通过评估后,他们发现维护这些重复代码的成本小于维护 #1 中的 JAR 或者 #2 中的 CommonAPI 服务,所以会继续让这些重复代码存在

假如某些 API 服务提供接口的出入参与后台服务的一致,此时该怎么办? 此时 API 服务的接口无须做任何事情,因为它只是一个简单的代理层

于是,有人提出:“每次一看到这些纯代理的 API 接口就不爽,能不能想办法把它们去掉。”办法倒是有几个,一起来看看

  • 网关直接绕过 API 服务调用后台服务,不过这样就会破坏分层,所以很快被否掉了
  • 在 API 服务层做一个拦截器,如果 URI 找不到对应 API 服务中的 controller mapping,就会直接通过 URI 找后台服务并进行调用。不过这种方式将大大增加系统的复杂度,出问题时调查起来更麻烦且收益不大。而写这些无脑代码不仅成本低,整体的接口列表还更可控

综合考虑后,最终决定保留无脑的代码

 

后台服务与 API 服务的开发团队如何进行分工?

专门的 API 开发团队负责 API 服务,而后台服务需要根据领域再划分小组的职责

这种划分方式的好处在于 API 团队能对所有服务有个整体认识,且不会出现后台服务划分不清晰、工作重复的情况。而坏处在于 API 团队整体业务逻辑偏简单,长久留不住人

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/vincent_wen0766/article/details/113839719

智能推荐

L293/L293d使用总结_l293d引脚图及功能-程序员宅基地

文章浏览阅读9.9k次,点赞5次,收藏20次。L293 L293d 应用总结_l293d引脚图及功能

innobackupex全量备份+增量备份手记_innobackupex 远程增量-程序员宅基地

文章浏览阅读862次。全量备份查库[root@localhost ~]# service mysqld5612 status[root@localhost ~]# mysql -uroot -prootmysql> select * from t1.test1;+------+| id |+------+| 1 || 2 || 3 |+------+3 rows in set (_innobackupex 远程增量

ConnectivityService框架初识-程序员宅基地

文章浏览阅读1.6w次,点赞10次,收藏96次。Android中提供的数据业务方式有几种:移动数据网络,WIFI,热点,网线等。这些数据业务本身可以独立使用,但是同一时刻,只能使用其中的一种数据业务方式。管理这些数据业务方式的使用由ConnectivityService,NetworkFactory,NetworkAgent,NetworkMonitor等来完成,ConnectivityService处于核心调度位置。ConnectivitySe_connectivityservice

Service Mesh 是什么,我们为什么需要它?_servicemesh性能-程序员宅基地

文章浏览阅读2.4k次。Service Mesh 是一个专门使服务与服务之间的通信变得安全、快速和可靠的的基础设施。如果你正在在构建一个云原生( Cloud Native )应用,那么你一定需要 Service Mesh 。在过去的一年中, Service Mesh 成为了云原生技术栈的关键组件。像 PayPal , Ticketmaster 和 Credit Karma 这样的大厂,已经将 Service Me..._servicemesh性能

xcode15出现大量Duplicate symbols_xcode 15 ld: 33 duplicate symbols-程序员宅基地

文章浏览阅读299次。新的连接器有诸多好处,尤其是对合并库的支持方面,具体可以查看WWDC 2023 SESSION 10268 Meet mergeable libraries.。然而,链接器的升级可能会出现不兼容老库的情况出现。从Other Linker Flags添加"-ld64"后,就会覆盖Xcode编译时选择的链接器,因此可以正常访问。升级到Xcode 15后,原先Xcode14可以编译的项目出现大量Duplicate symbols,且引用报错指向同一个路径(一般为Framework)下的同一个文件。_xcode 15 ld: 33 duplicate symbols

RuntimeError: “addmm_impl_cpu_“ not implemented for ‘Half‘_runtimeerror: "addmm_impl_cpu_" not implemented fo-程序员宅基地

文章浏览阅读2.6k次,点赞12次,收藏12次。将model.half() 改成 model.float()使用CPU进行推理时,遇到。_runtimeerror: "addmm_impl_cpu_" not implemented for 'half

随便推点

树莓派桌面远程访问_树莓派怎么远程桌面-程序员宅基地

文章浏览阅读909次。树莓派桌面远程访问方法一:VNC远程连接树莓派工具:VNC Viewer开启系统,在命令行输入:sudo raspi-config选择第五项Interfacing Options选择第三项VNC打开VNC Viewer工具,输入树莓派IP输入用户名:pi 密码:raspberry成功访问树莓派桌面问题解决:第一次进行上面操作的时候可能会遇到下面这样的情况在输入用户名和密码后,屏幕出现cannot currently show the desktop,百度查了一下。发现_树莓派怎么远程桌面

计算机视觉——【tensorflow入门】基于tensorflow框架的模型参数保存、重载及输出_tensorflow模型参数输出-程序员宅基地

文章浏览阅读1.3k次。基于tensorflow框架的模型参数保存、重载及输出编程语言:Python3.5框架采用:Tensorflow-gpu = 1.1.0在训练模型过程中不免会遇到需要存储模型参数的情况,在tensorflow框架下提供和Saver.save()函数来保存参数,保存的对象包括:权重及在程序中定义的变量,不包含图结构,保存的文件为checkpoint 文件。代码示例:变量..._tensorflow模型参数输出

re:从0开始的HTML学习之路 1. 标签间的关系与注释-程序员宅基地

文章浏览阅读347次,点赞12次,收藏11次。我们可以用记事本写一个如下的HTML文件,如

log4j、log4j2、slf4j、logback的使用_slf4j和log4j2依赖配置-程序员宅基地

文章浏览阅读126次。注意:slf4j + log4j2的适配器已经推荐使用 log4j-slf4j2-impl,而不是log4j-slf4j-impl。如果使用log4j-slf4j-impl,报如下错的话,说明你是slf4j 版本过高。_slf4j和log4j2依赖配置

注意力机制在图像分类中的表现与优化-程序员宅基地

文章浏览阅读395次,点赞5次,收藏6次。1.背景介绍图像分类是计算机视觉领域的一个重要任务,它涉及到将图像中的各种特征进行提取和分析,从而将图像分类到不同的类别。随着数据量的增加和计算能力的提升,深度学习技术在图像分类领域取得了显著的进展。在深度学习中,卷积神经网络(CNN)是最常用的模型,它可以自动学习图像的特征表达,并实现高度的分类准确率。然而,随着图像的分辨率和复杂性的增加,传统的CNN模型在处理大规模的图像数据集时存在一...

MySQL性能优化的参数简介-程序员宅基地

文章浏览阅读55次。 MySQL性能优化的参数简介公司网站访问量越来越大,MySQL自然成为瓶颈,因此最近我一直在研究 MySQL 的优化,第一步自然想到的是 MySQL 系统参数的优化,作为一个访问量很大的网站(日20万人次以上)的数据库系统,不可能指望 MySQL 默认的系统参数能够让 MySQL运行得非常顺畅。在Apache, PHP, MySQL的体系架构中,MySQL对于性能的影响最大,...

推荐文章

热门文章

相关标签