业务场景

曾经设计的一个供应链系统中,存在商品、销售订单、采购这三个服务,它们的主数据的部分结构如下所示

在设计这个供应链系统时,我们需要满足以下两个需求

  • 根据商品的型号/分类/生成年份/编码等查找订单;
  • 根据商品的型号/分类/生成年份/编码等查找采购订单

初期的方案是这样设计的:严格按照的微服务划分原则将商品相关的职责存放在商品系统中。因此,在查询订单与采购单时,如果查询字段包含商品字段,需要按照如下顺序进行查询

  • 先根据商品字段调用商品的服务,然后返回匹配的商品信息;
  • 在订单或采购单中,通过 IN 语句匹配商品 ID,再关联查询对应的单据

为了方便理解这个过程,画了一张订单查询流程图,如下图所示

初期方案设计完后,很快就遇到了一系列问题

  • 随着商品数量的增多,匹配的商品越来越多,于是订单服务中包含 IN 语句的查询效率越来越慢;
  • 商品作为一个核心服务,依赖它的服务越来越多,同时随着商品数据量的增长,商品服务已不堪重负,响应速度也变慢,还存在请求超时的情况;
  • 由于商品服务超时,相关服务处理请求经常失败

结果就是业务方每次查询订单或采购单时,只要带上了商品这个关键字,查询效率就会很慢而且老是失败。于是,重新想了一个新方案——数据冗余

数据冗余的方案

数据冗余的方案说白了就是在订单、采购单中保存一些商品字段信息。为了便于理解,下面借着上方的实际业务场景具体说明下,请注意观察两者之间的区别

调整架构方案后,每次查询时,就可以不再依赖商品服务了

但是,如果商品进行了更新,我们如何同步冗余的数据呢?在此分享 2 种解决办法

  • 每次更新商品时,先调用订单与采购服务,再更新商品的冗余数据
  • 每次更新商品时,先发布一条消息,订单与采购服务各自订阅这条消息后,再各自更新商品的冗余数据

如果商品服务每次更新商品都需要调用订单与采购服务,然后再更新冗余数据,则会出现以下两种问题

  • 数据一致性问题: 如果订单与采购的冗余数据更新失败了,整个操作都需要回滚。这时商品服务的开发人员肯定不乐意,因为冗余数据不是商品服务的核心需求,不能因为边缘流程阻断了自身的核心流程
  • 依赖问题: 从职责来说,商品服务应该只关注商品本身,但是现在商品还需要调用订单与采购服务。而且,依赖商品这个核心服务的服务实在是太多了,也就导致后续商品服务每次更新商品时,都需要调用更新订单冗余数据、更新采购冗余数据、更新门店库存冗余数据、更新运营冗余数据等一大堆服务。那么商品到底是下游服务还是上游服务?还能不能安心当底层核心服务?

因此,第一个解决办法直接被否决了,即采取的第二个解决办法——通过消息发布订阅的方案,因为它存在如下 2 点优势

  • 商品无须调用其他服务,它只需要关注自身逻辑即可,顶多多生成一条消息送到 MQ
  • 如果订单、采购等服务的更新冗余数据失败了,使用消息重试机制就可以了,最终能保证数据的一致性

此时,架构方案如下图所示

这个方案看起来已经挺完美了,而且市面上基本也是这么做的,不过该方案存在如下几个问题

1. 在这个方案中,仅仅保存冗余数据还远远不够,还需要将商品分类与生产批号的清单进行关联查询。也就是说,每个服务不只是订阅商品变更这一种消息,还需要订阅商品分类、商品生产批号变更等消息。下面请注意查看订单表结构的红色加粗部分内容

只是列举了一部分的结构,事实上,商品表中还有很多字段存在冗余,比如保修类型、包换类型等。为了更新这些冗余数据,采购服务与订单服务往往需要订阅近十种消息,因此,基本上需要把商品的一小半逻辑复制过来

2. 每个依赖的服务需要重复实现冗余数据更新同步的逻辑。前面讲了采购、订单及其他服务都需要依赖商品数据,因此每个服务需要将冗余数据的订阅、更新逻辑做一遍,最终重复的代码就会很多

3. MQ 消息类型太多了:联调时最麻烦的是 MQ 之间的联动,如果是接口联调还好说,因为调用哪个服务器的接口相对可控而且比较好追溯;如果是消息联调就比较麻烦,因为常常不知道某条消息被哪台服务节点消费了,为了让特定的服务器消费特定的消息,就需要临时改动双方的代码。不过联调完成后,经常忘了改回原代码

为此,我们不希望针对冗余数据这种非核心需求出现如此多的问题,最终决定使用一个特别的同步冗余数据方案

解耦业务逻辑的数据同步方案

解耦业务逻辑的数据同步方案的设计思路是这样的

  • 将商品及商品相关的一些表(比如分类表、生产批号表、保修类型、包换类型等)实时同步到需要依赖使用它们的服务的数据库,并且保持表结构不变;
  • 在查询采购、订单等服务时,直接关联同步过来的商品相关表;
  • 不允许采购、订单等服务修改商品相关表

此时,整个方案的架构如下图所示

以上方案就能轻松解决如下两个问题

  • 商品无须依赖其他服务,如果其他服务的冗余数据同步失败,它也不需要回滚自身的流程;
  • 采购、订单等服务无须关注冗余数据的同步

不过,该方案的“缺点”是增加了订单、采购等数据库的存储空间(因为增加了商品相关表)

仔细计算后,发现之前数据冗余的方案中每个订单都需要保存一份商品的冗余数据,假设订单总数是 N,商品总数是 M,而 N 一般远远大于 M。因此,在之前数据冗余的方案中,N 条订单就会产生 N 条商品的冗余数据。相比之下,解耦业务逻辑的数据同步方案更省空间,因为只增加了 M 条商品的数据

此时问题又来了,如何实时同步相关表的数据呢?直接找一个现成的开源中间件就可以了,不过它需要满足支持实时同步、支持增量同步、不用写业务逻辑、支持 MySQL 之间同步、活跃度高这五点要求

根据这五点要求,在市面上找了一圈,发现了 Canal、Debezium、DataX、Databus、Flinkx、Bifrost 这几款开源中间件,它们之间的区别如下表所示

从对比表中来看,比较贴近我们需求的开源中间件是 Bifrost,原因如下

  1. 它的界面管理不错;
  2. 它的架构比较简单,出现问题后,可以自行调查,之后就算作者不维护了也可以自我维护,相对比较可控。
  3. 作者更新活跃;
  4. 自带监控报警功能

因此,最终使用了 Bifrost 开源中间件,此时整个方案的架构如下图所示

上线效果

整个架构方案上线后,商品数据的同步还算比较稳定,此时商品服务的开发人员只需要关注自身逻辑,无须再关注使用数据的人。如果需要关联使用商品数据的订单,采购服务的开发人员也无须关注商品数据的同步问题,只需要在查询时加上关联语句即可,实现了双赢

然而,唯一让我们担心的是 Bifrost 不支持集群,没法保障高可用性。不过,到目前为止,它还没有出现宕机的情况,反而是那些部署多台节点负载均衡的后台服务常常会出现宕机

软件架构场景之—— 数据同步:如何解决微服务之间的数据依赖问题?相关推荐

  1. 软件架构场景之—— BFF:如何处理好微服务之间千丝万缕的关系?

    业务场景 之前设计的一个供应链系统中,它包含了商品.销售订单.加盟商.门店运营.门店工单等服务,涉及了各种用户角色,比如总部商品管理.总部门店管理.加盟商员工.门店人员等,而且每个部门的角色还会进行细 ...

  2. 微服务之间的最佳调用方式

    上一篇:3600万中国人在抖音"上清华" 0.2T架构师学习资料干货分享 茉莉花,别名:茉莉,拉丁文名:Jasminum sambac (L.) Ait,木犀科.素馨属直立或攀援灌 ...

  3. 微服务之间数据同步的思考

    周末无聊,来一篇服务之间数据同步的博客吧(主要讲注意的问题).具体什么业务场景就不举例了. ps:纯属个人瞎说,有错误.不足请大侠指出.嗯,开始说正事了. 业务流程 主要业务流程如下: #mermai ...

  4. Bifrost 同步数据库实现微服务跨库数据同步

    Bifrost Bifrost 可以在同步各种数据数据,类似于 Canal 当前支持的数据库 Redis MongoDB ClickHouse(DDL suppoted) MySQL(DDL supp ...

  5. Fegin拦截器解决各微服务之间数据下沉

    上篇说了当前端访问微服务网关,借助ZuulFilter过滤器来过滤所有请求,获取request,判断cookie是否有身份短令牌,request的header中是否有Jwt令牌,redis中是否有Jw ...

  6. 如何解决微服务的数据一致性分发问题

    介绍 系统架构微服务化以后,根据微服务独立数据源的思想,每个微服务一般具有各自独立的数据源,但是不同微服务之间难免需要通过数据分发来共享一些数据,这个就是微服务的数据分发问题.Netflix/Airb ...

  7. 波波老师: 解决微服务的数据一致性分发问题?

    点击▲关注 "中生代技术"   给公众号标星置顶 更多精彩技术内容 第一时间直达 介绍 系统架构微服务化以后,根据微服务独立数据源的思想,每个微服务一般具有各自独立的数据源,但是不 ...

  8. 如何解决微服务的数据一致性分发问题?

    介绍 系统架构微服务化以后,根据微服务独立数据源的思想,每个微服务一般具有各自独立的数据源,但是不同微服务之间难免需要通过数据分发来共享一些数据,这个就是微服务的数据分发问题.Netflix/Airb ...

  9. 一行代码就能解决微服务分布式事务问题,你知道GTS怎么做到的吗?

    2019独角兽企业重金招聘Python工程师标准>>> GTS直播火热报名中,直播直通车 一.GTS (Global Transaction Service)是啥? GTS(全局事务 ...

最新文章

  1. Node.js(nodejs)对本地JSON文件进行增、删、改、查操作(轻车熟路)
  2. (转载博文)VC++API速查
  3. vs2008添加excel类编译错误解决方法
  4. conky在ubuntu xfce4下面的配置
  5. [转载] 使用DirectInput进行交互
  6. 美丽的数学家:如果您讨厌数学,这些其实都是人生故事
  7. php打造自己的喜马拉雅,打造自己的私人知识宝库利器——mybase 7.3.5
  8. Exadata V2 Battery Replacement
  9. 微型计算机如何开声音,教你设置笔记本电脑上的杜比音效系统
  10. MySQL - 实战 棋牌游戏数据库开发
  11. 制作属于你的终端词典
  12. php微信摇一摇开发文档,摇一摇事件通知
  13. Ubuntu管理员密码
  14. 学生成绩管理系统(C语言)(链表)
  15. C-V2X 与智能车路协同技术 的深度融合
  16. 软件开发项目可行性分析报告样例
  17. python语言程序设计基础考试题库_中国大学MOOC(慕课)_Python语言程序设计基础_测试题及答案...
  18. vue2和vue3 面试题
  19. python 股票实时数据接口_获取股票实时数据的接口
  20. docker批量删除镜像-条件删除(过滤条件筛选)

热门文章

  1. 详解numpy中的array(附实例源码)
  2. Oracle数据库琐屑运用阅历六则
  3. xml文件拆分 python_使用Python解析大型拆分XML文件
  4. Nuxt 整合 element-tiptap 编辑器 上传图片到阿里云OSS关键方法
  5. (PD)PowerDesigner设计表时显示注释列Comment,Columns中没有Comment的解决办法(关联MySQL)
  6. 万能的应用商店_万能电影播放器
  7. 递推法之-------核电站问题(超简洁代码!!!)
  8. 电子商务模式之————B2C
  9. 下载steam创意工坊
  10. MTK 写入SN设置读不到SN问题