Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >业务侧最好的朋友:微服务中的 BFF 架构

业务侧最好的朋友:微服务中的 BFF 架构

作者头像
码猿技术专栏
发布于 2023-08-10 05:20:09
发布于 2023-08-10 05:20:09
3230
举报

大家好,我是不才陈某~

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

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

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

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

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

案例一

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

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

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

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

案例二

一个用户的提交操作常常需要修改多个服务数据,比如一个提交工单的操作,我们需要修改库存、销售订单状态、工单等数据。关注公众号:码猿技术专栏,回复关键词:1111 获取阿里内部Java性能调优手册!

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

通过上图,我们发现服务间的依赖问题给技术迭代带来了地狱般的体验,讲解,这里就不过多赘述。

为了解决这 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 团队整体业务逻辑偏简单,长久留不住人。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2023-07-03,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 码猿技术专栏 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
微服务架构中的BFF到底是啥?
在《技术中台与业务中台都是啥玩意》一文中留下一个问题:BFF是啥?为啥在API网关和业务中台之间加入了一层BFF?考虑到在实际工作中,我的大部分同事都问过这个问题,这里我也总结一下进行答复。
Edison Zhou
2020/07/13
3.1K0
微服务架构中的BFF到底是啥?
微服务架构~BFF和网关是如何演化出来的
BFF(Backend for Frontend)和网关Gateway是微服务架构中的两个重要概念,这两个概念相对比较新,有些开发人员甚至是架构师都不甚理解。
猿天地
2019/11/26
1K0
微服务下使用GraphQL构建BFF | 洞见
微服务架构,这个在几年前还算比较前卫的技术在如今遍地开花。得益于开源社区的支持,我们可以轻松地利用 Spring Cloud 以及 Docker 容器化快速搭建一个微服务架构的原型。不管是成熟的互联网公司、创业公司还是个人开发者,对于微服务架构的接纳程度都相当高,微服务架构的广泛应用也自然促进了技术本身更好的发展以及更多的实践。本文将结合项目实践,剖析在微服务的背景下,如何通过前后端分离的方式开发移动应用。 对于微服务本身,我们可以参考 Martin Fowler 对 Microservice 的阐述。简单
ThoughtWorks
2018/04/17
2.1K0
微服务下使用GraphQL构建BFF | 洞见
架构与微服务本质论
为应对如今无线优先和全渠道用户体验的需求和挑战,我们该如何设计灵活的面向体验的微服务架构?它有哪些模式和最佳实践?携程,Netflix和SoundCloud这些知名互联网公司是如何实践面向体验的微服务架构的?在过去的时间,大牛马丁福勒对微服务有哪些新的观点?
技术zhai
2019/02/15
1K0
介绍一个架构新词-BFF(这个和微服务也有关系)
本篇介绍一个概念BFF,Backend-For-Frontend,结合几篇参考文章,从不同的角度理解BFF的发展来源和实际作用。
needrunning
2019/09/10
8.8K1
介绍一个架构新词-BFF(这个和微服务也有关系)
微服务架构下的核心话题 (二):微服务架构的设计原则和核心话题
     毫无疑问,微服务架构的设计原则和核心话题是本文要讨论的重点,也是打算从零基础开始构建微服务架构需要事先考虑、规划的。一个好的产品、应用能否稳定运行,持续开发,满足业务需求,能否经得起现实的考验,就需要在设计阶段考虑很多、很多,以确保它的健壮性。
xcbeyond
2020/01/21
7740
微服务架构下的核心话题 (二):微服务架构的设计原则和核心话题
有赞零售中台建设方法的探索与实践
上图是有赞零售SaaS业务整体业务架构概览,大体上可以分为前台业务、中台业务、后台业务。
有赞coder
2020/08/24
1.1K0
有赞零售中台建设方法的探索与实践
B站基于微服务的API网关从0到1的演进之路
如果你在 2015 年就使用 B 站,那么你一定不会忘记那一年 B 站工作日选择性崩溃,周末必然性崩溃的一段时间。
JackJiang
2022/06/14
9810
B站基于微服务的API网关从0到1的演进之路
微服务架构实施原理
作者:风中程序猿,来自:cnblogs.com/fangfuhai 1 题记 基于微服务架构和Docker容器技术的PaaS云平台建设目标是给我们的开发人员提供一套服务快速开发、部署、运维管理、持续开发、持续集成的流程。 平台提供基础设施、中间件、数据服务、云服务器等资源,开发人员只需要开发业务代码并提交到平台代码库,做一些必要的配置,系统会自动构建、部署,实现应用的敏捷开发、快速迭代。 在系统架构上,PaaS云平台主要分为微服务架构、Docker容器技术、DveOps三部分,这篇文章重点介绍微服务架构的实
架构师小秘圈
2018/06/04
1.6K0
什么是微服务
在介绍微服务时,首先得先理解什么是微服务,顾名思义,微服务得从两个方面去理解,什么是"微"、什么是"服务", 微 狭义来讲就是体积小、著名的"2 pizza 团队"很好的诠释了这一解释(2 pizza 团队最早是亚马逊 CEO Bezos提出来的,意思是说单个服务的设计,所有参与人从设计、开发、测试、运维所有人加起来 只需要2个披萨就够了 )。 而所谓服务,一定要区别于系统,服务一个或者一组相对较小且独立的功能单元,是用户可以感知最小功能集。
前朝楚水
2018/07/26
9570
「系统架构」微服务探究之初识微服务
在传统的开发中,我们通常是将所有的功能打包在一起,然后统一部署在一个JEE容器(Tomcat,JBoss,WebLogic)里,包含了 DO/DAO,Service,UI等所有逻辑。如下图所示:
ZhangXianSheng
2019/05/28
3740
BFF 架构简介
BFF全称是Backends For Frontends(服务于前端的后端),Sam Newman曾在他的博客中写了一篇相关的文章——Pattern: Backends For Frontends,在文章中Sam Newman详细地说明了BFF。
JavaEdge
2021/02/23
1.3K0
BFF 架构简介
微服务架构:搭建网站扫码登录的功能设计
微信扫码登录大家都是应用比较多的登录方式了,现在大的购物网站像京东、淘宝等都支持使用APP扫码登录网站了。今天就用APP扫码登录网站的实例来举例说明微服务架构的搭建过程。
搜云库技术团队
2019/10/17
1.7K0
微服务架构核心基础讲解[入门级]
Martin(微服务提出者也叫 Martin)刚来到公司时是一个基层员工,它上面有经理、老板,那个时候所有人都听老板的指挥。但是过了两年,公司的人越来越多,原来的模式下整个公司的运作效率太低,管理也很混乱。
码农小胖哥
2020/06/11
6160
微服务架构核心基础讲解[入门级]
对微服务架构设计实践中若干问题的探讨
今天讨论下在微服务架构实践中经常遇到的一些问题的思考,其中有些来源于我们自己的微服务改造项目,有些来源于客户现场微服务架构实施项目和售前方案沟通。
xcbeyond
2020/11/03
5100
对微服务架构设计实践中若干问题的探讨
对微服务架构设计实践中若干问题的探讨
今天讨论下在微服务架构实践中经常遇到的一些问题的思考,其中有些来源于我们自己的微服务改造项目,有些来源于客户现场微服务架构实施项目和售前方案沟通。
xcbeyond
2021/01/26
4590
对微服务架构设计实践中若干问题的探讨
SpringCloud简介与微服务架构
微服务架构(Microservice Architecture)是一种架构概念,旨在通过将功能分解到各个离散的服务中以实现对解决方案的解耦。你可以将其看作是在架构层次而非获取服务的类上应用很多SOLID原则。微服务架构是个很有趣的概念,它的主要作用是将功能分解到离散的各个服务当中,从而降低系统的耦合性,并提供更加灵活的服务支持。
BUG弄潮儿
2021/02/03
6250
SpringCloud简介与微服务架构
电子合同在食品行业的应用方案
据了解,一家大型食品组织,一年要处理各类内、外部用章请求达3-7万次。引入电子合同简化用章、签约成本,已经成了许多组织进一步释放管理动力的新选择。
契约锁电子合同
2020/06/11
1.9K0
电子合同在食品行业的应用方案
微服务--限流
在上一篇文章中我们讲解了微服务中的熔断处理方式,在实际开发中和熔断相配的技术就是限流,那么这篇文章我们将着重讲解限流的的知识。
喵叔
2022/09/28
4160
微服务安全认证架构是如何演进而来的?
之前有同事问为何要用基于JWT令牌的认证架构,然后近期又有童鞋在后台留言问微服务安全认证架构的实践,因此我决定花两篇推文来解答一下。为了答好这个话题,我们先来看看微服务的安全认证架构是如何演进而来的,从而更好地理解。
Edison Zhou
2020/07/25
4350
微服务安全认证架构是如何演进而来的?
相关推荐
微服务架构中的BFF到底是啥?
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文