一个有线下门店业务的商家,在做业务扩张时,考虑到扩张的成本,会寻找一套可复制的经营方式,通过连锁的模式进行规模化扩张。
不同行业对于规模化的应用也有所不同:对于零售商家,单个门店的人流量和仓储能力都是有限的,更多的门店往往意味着更大的人流和更快的物流;而对于美业这样重线下服务的商家,单个门店能覆盖的服务范围是有限的,更多的门店意味着更大的服务范围和更多的客户;即便对于单纯做网店的商家,也可以利用合伙人的能力来帮助自己更快的发展。
在从单个门店到多个门店的过程中,商家的管理和运营都会变得更加复杂,怎么对门店的商品、店铺装修等进行统一管理,怎么发挥总部的优势给底下的门店带流量,怎么去激励业绩不好的门店等等。对应的在技术侧,架构也需要演进、迭代来支撑越来越复杂的业务,从前只需要考虑一个门店,现在需要考虑多个,考虑品牌下总部和门店、门店和门店间的关系等等。
以上是对连锁业务复杂度由来的简单介绍。复杂业务如果没有得到相应的设计和处理,就会带来痛点,这篇文章聚焦在其中一个比较常见的痛点上,聊聊这个痛点是如何产生,以及我们如何解决。
连锁的很多功能都需要深入到具体的业务去设计和开发,在给一个业务去迭代一个连锁的能力时,都需要对这块业务原本的逻辑有一个深入和全面的理解后才能上手,比如商品库、跨店核销、组合卡适用多门店等等。
经历了这么多项目后,会发现有一些共同的痛点: 1、系统不易理解:开发同学往往会发现原有系统设计和代码实现逻辑混乱,不知从何下手 2、代码不好修改:调整一个连锁的逻辑,会影响到原有一些很底层的功能,影响面很大 3、维护和部署不方便:改动一个连锁的功能,需要发布核心应用
整体而言就是在涉及到连锁解决方案的需求时,研发的生产力会比较低。
于是就在想,有没有办法通过一些优化和设计,来让连锁的开发同学: 1、更快速的理解一个业务,做出合理的技术设计 2、更舒服的编写代码 3、更安全的发布一个新功能
在探索和实践这件事情时,大致的思路是这样: 1、找一个具体的连锁场景,分析其业务逻辑:这里我找的典型场景是商品,考虑到大家对商品都比较了解; 2、分析下现在是怎么实现的? 3、现在这样的实现方式,会有哪些问题? 4、怎么去优化它?
这里用一个多维表格,描述下一个简单的 B 端商品业务流程,包含:商品创建、上下架、查看和编辑。
横轴是业务场景,纵轴是组织类型,中间的内容,是这些业务场景在对应的组织类型上会有什么样的表现: 注: 1、横向越权控制,是指总部只能操作总部下的门店,而不能操作归属于其他总部的门店 2、总部编辑商品,信息同步至门店时会有一定的同步规则,不是所有信息都会同步过去,这里不展开细讲,知道有这样一个逻辑即可 3、同样的,对于从总部上架过来的商品,门店只允许更新部分属性,这些都属于连锁经营场景下的特有逻辑
以编辑商品为例,现在的实现大致分两步: 1、更新商品,发送商品变更消息 2、消费者收到消息,处理变更信息,更新分店商品看起来是很清晰,但实现上还是有些问题,下面是伪代码:
编辑商品
updateItem(req) {
// 前置校验:参数校验、权限校验、编码重复性校验等等
validate(...);
// 如果是总部更新
if(updateByHeadQuarter(...)) {
// 更新总部商品
...
}
// 如果是门店更新
if(updateByStore(...)) {
// 只允许更新部分属性
...
}
// 发消息
sendMsg(...);
}
消费者监听消息,更新门店商品信息
handle(msg) {
// 根据不同的商品事件类型,走不同的处理动作
switch (msg.goodsEventType) {
// 新建商品
case NEW:
...
// 更新商品
case UPDATE:
processUpdateEvent(msg);
// 上下架
case TAKE_UP_DOWN:
...
// 其他事件类型,不一一列举
case XXX:
...
}
}
processUpdateEvent(msg) {
// 如果是总部更新,则同步信息到门店
if (updateByHeadQuarter()) {
// 查询总部下所有门店
getAllStore(...);
// 更新这些门店的商品信息
updateAllStore(...);
}
}
刚刚提到的所有业务功能,都是由商品服务提供的,把他们都平铺出来,就是这样: 一个直观的感受就是这个商品服务做了太多事情,既有一些商品通用的底层能力,又有连锁的逻辑。
讲具体些,以上面的商品编辑为例,当前的实现存在的问题有:
1、耦合 通用的底层逻辑和连锁逻辑耦合在一起,比如门店能更新哪些商品属性,这个逻辑和通用的更新商品逻辑耦合在了一起,如果后续需要调整这个逻辑,比如允许门店自己修改商品条码,那就会动到通用编辑商品的代码;
2、忙碌的消费者 代码里通过一个消费者,来监听和处理所有的商品事件,而把总部商品同步到门店,这本身是一个独立的逻辑,却被揉在了这个消费者里去实现,本质上是因为大家对这个服务的定位有关,大家觉得这个服务就是要把所有的商品能力都实现了,所以就会写出这样一个忙碌的消费者;
3、能力缺少复用 无论是编辑单个商品,还是批量编辑门店商品,逻辑上应该是一致的,但是我们却看到现在是两份代码,代码没有任何复用;
基于以上分析,再结合一些常用的设计模式和原则,于是有了以下的优化思路:
1、开闭原则(OCP) 能不能让允许门店更新哪些属性,和商品通用编辑能力隔离、解耦,不互相干扰;
2、单一职责(SRP) 能不能把「总部商品信息同步到门店」,这个逻辑单独抽离,独立成一个消费者,就只专心做这一件事情;
3、让商品更专注于自己的领域 能不能让商品不太去关注总部、门店这样的连锁组织关系,让它去提供一套对所有店铺通用的商品能力,和店铺的耦合,就只是弱耦合一个店铺ID。
基于上面的「业务场景-组织关系」的二维矩阵,我们发现可以把业务场景做进一步的抽象和拆分,分成: 单店能力 + 连锁能力
划分的标准很简单,如果一个能力是单店商家就需要有的,那就属于单店能力,如果是连锁商家(两个或两个以上门店的商家)才需要的能力,那就属于连锁能力。
我们来看看如果按照上面的划分标准,之前的场景可以抽象出哪些能力,这些能力属于单店能力还是连锁能力:
简单描述下: 1、总部下创建商品,其实就是在一个店铺下去创建商品,明显是单店能力; 2、总部把商品上架到门店,商品上架,本身属于单店能力,但是针对总部对门店的横向越权控制,这个属于连锁的范畴; 3、同样的,编辑商品、下架商品,这些也是单店能力,但是编辑总部商品后,信息怎么同步给门店,门店可以编辑哪些商品属性,这些则是连锁场景下才需要的能力,属于连锁的范畴。完成这样的结构化分析后,我们再把之前的场景进行一个拆分:
虽然只是给一团浆糊来一刀,变成两团浆糊,但是会发现这样的划分,能帮助我们更好的思考,并且基于这两团浆糊,做出更多的设计和优化。
把商品能力拆解为连锁能力和单店能力后,在部署方式上也进行了对应的设计,来支持业务的快速发展和系统的稳定: 商品的单店能力由 mei-goods 实现,连锁能力交给 mei-chain 负责。当然如果单店和连锁能力都需要再往细拆分,也可以拆分成更细的服务。
带来的好处是: 1、代码解耦 比如门店能更新哪些商品属性,这个属于连锁能力,需求有调整时只需修改发布 mei-chain,不会修改 mei-goods,不影响单店逻辑; 2、部署隔离 就算 mei-chain 挂了,也不影响单店操作场景,只影响连锁场景。 举个例子,如果 mei-chain 所有实例都下线了,那么总部和门店还是可以创建商品,但是如果总部编辑了商品,同步信息给门店时会失败,等 mei-chain 恢复后重新启动消费者,把消息消费了,业务和数据即可恢复正常; 3、职责单一明确 通过这样的部署方式,促使开发时会进行更细粒度的思考,比如总部-门店商品信息同步的事情属于连锁能力,就该由 mei-chain 负责,通过单独一个消费者实现,而不是在 mei-goods 写一个庞大的消费者。
图中画了一个 api 层,这里想展开聊一下。
其实本来 api 层我画的是虚线,因为 mei-goods 和 mei-chain 本身都可以对外提供 api 接口,不管是 rpc 还是 restful,都可以提供,这样 api 层也就只是一个概念意义上的存在了。
之所以画成实线,一方面是因为对于前端来说 api 层是很真实的存在,前端同学不关心接口底层是由 mei-goods 还是 mei-chain 实现,只对接 api;另一方面,我们可以在 api 层做一些聚合类的操作,比如对于商详页,我们除了基础的商品信息,还需要聚合一些活动信息,比如这个商品是不是参与了秒杀、拼团、体验价,还需要聚合库存等等,如果没有单独拎出来,那我们就很容易把这些逻辑也放到 mei-goods 或者 mei-chain 来实现,这样就又造成了污染。
我们可以提供一些大而全的接口,但是我们要基于一些小而精的接口来提供,这种思路是和最近常常被提起的 Serverless 背后的理念是一致的。每个领域都会单独提供自己领域内的接口,然后再在这上面做一些胶水层,进行数据的聚合,现在我们可能是在后端应用做,以后我们可以在 node 层做,甚至可以在 Serverless 架构中的云函数里做。
常规来说,连锁逻辑(mei-chain)由连锁团队开发,单店逻辑(mei-goods)由营销、店务团队开发。
实际情况常常是资源紧缺的,比如营销店务同学业务很忙,则连锁团队即开发 mei-chain 又开发 mei-goods,反过来,连锁同学很忙,那营销店务同学也可能即开发 mei-chain 又开发 mei-goods。
其实架构的设计终究还是为业务服务的,由哪个团队、由谁来开发哪个应用,这个并不是特别重要,重要的是通过这样一种架构的设计,我们能够: 1、快速支撑业务的调整和发展 2、提高系统稳定性 3、引导开发和产品思考
最后一点似乎有点狂,一个架构的设计还能引导产品思考?其实这点也是在组内分享的时候,产品同学的共鸣给我带来的启发。通过对连锁和单店的解耦,开发同学会自然的将连锁、单店的逻辑分开考虑,不耦合在一起,而开发同学在和产品同学的频繁接触过程中,会反过来去推动产品,也这样去思考问题。
正如 Bob 大叔在《架构整洁之道》里说的,软件架构的设计,会影响软件系统的全生命周期:
软件架构设计的主要目标,是支撑软件系统的全生命周期,设计良好的架构可以让系统便于理解、易于修改、方便维护,并且能轻松部署。 软件架构设计的终极目标,是最大化程序员的生产力,同时最小化系统的总运营成本。 —— 《架构整洁之道》
我们在小程序直播项目里第一次采用了这样的设计。
对于直播而言,理论上每个具有经营能力的组织都可以发起直播,但是有能力去把直播这件事做好的往往只有总部,所以我们既要支持有能力直播的门店自己发起直播,又要让总部可以给门店去直播带货。很自然的我们把创建直播间这类能力归为单店能力,把和总部统一直播相关的,诸如批量应用直播间、批量配置直播间浮窗等,归为连锁能力。
实践中印象比较深的,是由于进行了拆分,后端给前端提供了更低粒度的接口,并且这次后端也没有在 api 层进行聚合,有些业务需要前端去进行聚合,这和以往一个接口对应一个页面的前后端配合模式不太一样,也算是一种尝试吧。
这也是在组内分享时大家讨论的最深的一个问题。
在思考这个问题前,我觉得需要先思考另一个问题 —— 店铺域和其他领域的关系。
每个领域都有自己负责的那块业务,这些业务大多都会关联一个店铺ID,虽说对于一件商品、一个营销活动、一个客户,它们本身并不一定需要和店铺有什么关系,但是考虑到实际的电商经营场景,这些商品、营销活动终究要归属到某个店铺上。
但是其他领域在进行设计的时候,出于聚焦也好,出于减轻思考压力也好,并不会把店铺摆在台面上来讨论,因为我们默认一件商品、一个营销活动、一个客户天然就是要挂在一个店铺上的。我们更多的是聚焦在商品、营销活动、客户域自身应该有哪些业务属性和动作。
而这个时候如果把一些连锁的场景考虑进来,就会让思考变得复杂,这时候我们可以做一个减法,只考虑单门店商家,无需关心操作的是总部,还是门店,还是合伙人,还是其他店铺角色。从业务的角度思考,连锁商家都是从单门店做起来的,在探索和实践出一套可复制的经营之道后再进行规模化,把整套方法应用在每个门店上,对应到技术上,最后的操作还是要落在每一个门店上的,在基于单门店场景构建单店能力后,我们同样可以继续基于这套单店能力,去继续搭建更复杂的连锁能力,而这个时候,我们只需要去思考连锁场景,因为单店的我们已经解决了。
回到问题本身,DDD(Domain Driven Design,领域驱动设计) 是一套解决复杂业务问题的方法,对于已经很复杂的业务,如果我们同时考虑单门店和连锁的场景,就会让业务变得更加复杂,加大了 DDD 实践起来的难度,这时候我们对业务场景划分了边界,先聚焦在单门店的场景,再聚焦在连锁的场景,把他们分成两个独立的领域去解决,比如在这个例子里,我们把商品分成单店商品领域和连锁商品领域,他们合起来共同为有赞的商品领域提供解决方案。
最后想探讨一下这套设计的局限性。
有一些业务需求在单门店商家里是不存在的,比如绝大多数的供应链场景、财务场景,比如库存调拨、门店要货、资金分润等等,这些场景只会发生在连锁的商家,那也就不需要像上面那样按照单店、连锁的维度进行划分了。
这篇文章主要介绍了连锁业务的痛点,并且分析了痛点的来源,以及通过结构化分析进行解耦的探索和实践,中间也穿插了一些自己对于架构设计、DDD、Serverless 的一些思考和看法。希望通过这篇文章,能为读者在解决复杂业务上提供一些参考价值。
本文由哈喽比特于4年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/yHr8itt2GN7YBt4gBa64Kg
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。