简介: 当我们在学习DDD的过程中,感觉学而不得的时候,可能会问:我们还要学么?这的确引人深思。本文基于工作经验,尝试谈谈对DDD的一些理解。
(资料图片)
作者 | 闵大为
来源 | 阿里开发者公众号
《阿甘正传》中,阿甘开始了不停地跑步,一段时间后,后面就有了很多追随者一起跑,他们为什么跑哪?
阿甘:我也不知道,只是想跑而已。 追随者:感觉这样做是有意义的,而且阿甘也还在前面领跑。类似地,一开始我也不知道DDD是什么,但当发现大家都在提DDD、都在学DDD的时候,我也像跟跑者一样不由自主地加入了前行:既然有大牛提出了DDD,既然那么多人趋之若鹜,那么肯定有可取的地方。
然而,有一天,阿甘停止了跑步,他不想跑了,追随者遇到了一个问题:我们还要跑么?当我们在学习DDD的过程中,感觉学而不得的时候,可能也会问:我们还要学么?这的确引人深思。
本文基于工作经验,尝试谈谈对DDD的一些理解,希望能够更好地探寻学习DDD的意义。
无论做业务,还是做平台、中台,大家常常会被交错复杂的业务逻辑、晦涩耦合的业务代码搞得心力憔悴。我想,大家对DDD的追求,也是对轻松支撑业务发展的诉求,在探寻有没有合适的理论可以改善现状。毕竟,美好生活,共同向往。
我们选择各种框架、进行各种组织设计,核心是为了提高生产效率。但如果业务逻辑都是 case by case 地进行实现、缺少复用,那么研发成本是非常高的、投入周期也会非常长。
为了增加复用、缩短业务的落地时间,就需要很多通用的能力、产品。在我们的交付过程中,主要有两个层次:
基础能力 :相对原子的能力是基础(域)能力,这个可以较好地支持业务定制。由于比较基础,表达的产品能力范围也是很大的。但是,一个完善的产品需要串联的基础能力是非常多的,串联的成本也是非常高的。 平台产品 :基础能力的通用性,意味着缺少对场景的理解,缺少了进一步提升生产效率的“基因”。所以在交付的时候,会基于一些高频场景进行抽象,形成平台的产品能力,争取做到“拆箱即用”。业务基于“平台产品”这层进行定制的时候,理解成本会大大减少。这样的层次,看上去很美好:在起步阶段,由于缺少历史包袱,的确可以提升一定的生产效率,这是能力本身的收益。但是,越往后,随着业务接入的增多,业务之间开始互相影响,研发的阻力也越来越大。
研发效能降低的重要原因在于:更多的时候,我们还是按照“业务能跑起来,怎么快怎么来“的逻辑去做相关工作,遇水搭桥,遇山开洞,然后直达目的地,进行信息的传达、数据库字段的操作。
这样的过程,违背了我们”希望通过业务场景,丰富平台能力,同时保证内核干净“的初衷。能力应该是基于相对多的用例、相对完善的思考进行抽象,是横向统一看,有更深刻的理解,但是垂直的交付,让我们更加纵向地处理问题,往往只是“窥探”了链路,在交付时长和业务节点的限制下,很难想得更加全面、深刻,难以做出更通用的设计。
那么,为什么关注DDD?如果说DDD直击了软件复杂度的核心——“问题域”,那可能还是比较抽象。具体来说,因为这的确符合我们追求的价值观 【提升长期的生产效率】 :
细分领域,培养专业的人、事 :因为DDD的核心是要求让各个领域做好理解和封装,把一个业务需求拆解、安放在各自合理的地方,通过这样的分解与沉淀,保证了领域的输入,能够得到长期可持续的发展,形成竞争力。 机制保障,不依赖易变的事物 :DDD其实在总结很多通用的技巧和经验,能够让这样的实施更具有确定性。无论是聚合根对领域实体的管控能力、限界上下文的交互策略、领域内核的抽象地位...等等,一旦选择尊奉,确定下来,就能够落在代码结构、组织关系、团队文档中,形成共识,不会因为人员等因素的变化而剧烈波动。对DDD的关注,可以类比于我们对“工匠精神”的关注,对DDD的重视,也是我们对业务理解的重视。 贴近业务,更要理解业务,不仅要理解业务,更需要理解大多数的业务。 这样的追求,让我们往上看了一个层次,回归了最本质的问题:我们要解决什么问题?如何能够解决得更好?
不知道大家是否有这样困惑:DDD的学习过程好像是”大海捞针“的过程。即使能够捞到点东西,使用起来,还是会有种“东施效颦”的感觉,并不是很自然。为什么学习DDD那么困难那?
论语中的下面这个场景,和我们的困惑是比较相似的:
叔孙武叔语大夫于朝曰:“子贡贤于仲尼。” 子服景伯以告子贡,子贡曰:“譬之宫墙,赐之墙也及肩,窥见室家之好;夫子之墙数仞,不得其门而入,不见宗庙之美、百官之富。得其门者或寡矣,夫子之云不亦宜乎!”
正如要感受到孔子达到的境界,自己的学问也需要有一定的积累。我们要感受到DDD的力量,自己本身就要成长到一定程度(如:经历了一些成功或者失败的设计,有自己的经验或者教训),才能形成共鸣和认同。
工作中,的确很少看到DDD的最佳实践。在复杂的业务面前,谁也没有勇气说,哪个软件结构是理想的设计:
因为这不是一个确定性的问题分解,你的设计会被放在显微镜下研究,总能找到各种反例。 而且,我们深知,最佳的实践,一定是做得足够的“软”,对扩展留有设计,能够随着业务发展而迭代,不是一个静态的结果。因此,打开学习的大门不是几个案例就能一蹴而就的,需要结合我们自己的工作,慢慢累积、体会。
我们有一种困惑,到底怎么样算是DDD:
实践的个例,难以信服 :当我们看到”DDD实践“的时候,可能会发问:这也算DDD?不就是一个正常的服务端框架与方案,也无法涵盖其它的场景或者部门系统。 抽象的理论,觉得空洞 :当我们看到”抽象的DDD定义与策略“的时候,可能会发问:这也算DDD?不就是一些软件设计的共识,然后强加了一些名词定义,有些策略与我们手上的系统也并不匹配。无论往抽象看,还是往具体实现看,都很难找到令人信服的理论与实践(能够有确定性的落地能力)。 因为这不像23个设计模式那样,可以通过N个模版就能涵盖大多数的模式。
为什么不能产生特定的模式呢?可以结合下图进一步来看:
抽象理论 :如同抽象的接口一样,"DDD理解"最上面的学习主要是理论定义,比如:聚合根、值对象、资源库、核心域、支撑域等各种定义,这些是易于理解掌握的。 通用实践 :如同相对具体的抽象类一样,"DDD理解"中间层次是一些通用原则和技巧,比如:上下文的映射策略、架构的选择等。这些因素是确定的,但需要自主进行取舍与选择,并且需要与时俱进,增量的部分需要你自己学习补充。 具体实践 :如同具体的类实例一样,"DDD理解"中下面层次是一系列的具体实践,结合各自的业务场景,进行了不同因素的设计、取舍与补充。因为涉及的选项较多,造成最终的选择结果往往是发散的,令人感觉“千人千面”。两者不同的地方是:
“代码抽象层次”中层次关系是比较明确的,且有约束。 “领域驱动理解层次”中无法提供明确的约束,都是多个策略的取舍、一些关键的建议。因此,由于问题的抽象层次较高,各种策略的不确定性较高,很难在DDD中产生像”23个设计模式“那样精炼的模式。一定要有的话,也是一系列的模式,比较发散。
DDD理解层次的类比
因此,我们渐渐明白:DDD面向的是软件的“软”,涵盖方方面面。DDD的深入理解,需要“千锤百炼”后,才能明白那些深入简出的建议,才能体会那句“师傅领进门,修行靠个人”的真言,才能感受到“众里寻他千百度,那人却在灯火阑珊处”的美妙瞬间。
虽然DDD本身的实践可能千人千面,但是一些核心主题的思考应该是聚焦的,这些高频主题的理解,能让我们更好地进行设计,讨论的性价比也是较高的。接下来,会基于“6大设计原则”(solid 原则)为引子,去看看DDD中的一些关键理解。
单一职责是说:一个类应该只有一个发生变化的原因。职责的单一,可以更好地内聚,减少耦合,方便演化。
DDD里面的领域划分可以类比思考。对领域的划分,无论是按领域实体,还是按照功能模块,还是按照服务等划分,其实都想尽量保证领域的正交,能够独立演化和发展。
领域怎么切分比较合适?刚进入业务平台的时候,了解到领域切分是按“一个或多个实体对象”的边界来切分。这的确比较合理,因为领域的核心职责就是对领域实体进行管理。但这是果,还是因?在切分的时候,是因为我们有了对领域的判断,所以某些实体被分在一起比较合适;还是因为某些实体有明显边界,所以可以形成一个领域?就比如下面的图:
可以整体作为1个部分。 可以按竖着的正、负切分2个部分:上面1个(红),下面2个(黄、绿)。 可以按横轴的正、负切分2个部分:左边1个(黄),右边2个(红、绿)。 可以切分成3个部分(红、黄、绿)。聚类的例子
这的确引人深思。切分比较容易的时候,往往是因为已经有了行业的标准(如:电商系统有订单、支付、物流、库存等领域是比较合理的)。那行业的标准来自哪里?是来自于演化:
开始的时候,可能只是一个大交易,比如:支付开始的时候只是买卖双方自己协议,也不需要建模。 后面支付发展了,也就独立出来一个域。原来不需要专人维护,后面会渐渐拉出一个团队来承接相关研发。所以,领域的演化和划分,很类似“启发式算法”(一个基于直观或经验构造的算法,在可接受的花费下给出待解决组合优化问题每一个实例的一个可行解):
初始化 :按照的经验初步的划分,也可以是行业标准(没有行业标准的时候,就只能靠经验了)。 花费评价 :生产、交付过程的人力成本度量,关注理解成本、开发效率、系统稳定性、运维成本等因素。 更优解 :在业务发展过程中,计算花费评价,分析影响评价的“好因素”、“坏因素”,进行进一步调整。往往到最后,我们会发现:
调整的内容 :其实是匹配生产关系。 调整的原则 :追求职责的内聚,精细化分工。 不断调整的原因 :业务在发展,内聚的标准需也要与时俱进。此外,从关联的角度看,往往我们看组织架构,就能看到领域的边界,核心原因还是组织架构也是要适应生产关系,follow更优解的结构,是相辅相成的,也就能互相窥探。
点击查看原文,获取更多福利!
https://developer.aliyun.com/article/1150965?utm_content=g_1000368227
版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。