本文书接上回《学习真DDD的最佳路径》,关注公众号(老肖想当外语大佬)获取信息:

  1. 最新文章更新;

  2. DDD框架源码(.NET、Java双平台);

  3. 加群畅聊,建模分析、技术实现交流;

  4. 视频和直播在B站。

神秘的“凭经验”

一千个人眼中有一千个哈姆雷特,每个人的经历不同,认知不同,那么看待哈姆雷特的角度和感受也不同。在软件工程领域,也有著名的关于如何做好软件设计的观点:“凭经验”。然而,“凭经验”就意味着不可复制,如果软件设计只能凭经验,那还不如说是撞大运。

读过本系列前文《老肖的领域驱动设计之路》的朋友,应该知道,我对此是持相反的观点的,今天通过几分钟的篇幅,呈现我的理解,尝试破除凭经验的魔咒。

凭经验与主观判断

首先,我认为依靠经验来决策,本质上是一种主观判断,尤其是那些无法由客观事实和逻辑推导的判断和无法建立普遍共识的判断。至少凭经验这个方法本身,是主观的。试想一下,如果人们都仅仅依靠自己的过往经验来做判断,那么“靠左走好”和“靠右走好”就会成为普遍的观点冲突,而这个冲突的背后就是,“我不要你觉得,我要我觉得”。

当然更准确的说法是“凭经验”意味着不能绝对客观。

主观与客观

主观的反面是客观,客观意味着观点基于客观事实,基于符合事实的推导逻辑链,也就意味着客观决策,能够在一定的规则范围内建立共识的判断结果,那么我们可以得到如下观点:

  1. 主观判断对应非标,无法标准化

  2. 客观判断对应标准,可以标准化

如果对于一件事情可以标准化判断,是不是意味着这件事是可执行、可落地的?

需求分析和建模设计中的主观与客观

先举个例子,假设我们有个需求:“整理房间里的物品”,那么我们的解决方案可以是:“取N个收纳箱,把物品一个个放进去”,那么这里就有一些主观和客观的决策在里面。

主观的部分有:

  1. 决定用多少个收纳箱

  2. 哪些物品放哪个收纳箱

客观的部分有:

  1. 收纳箱之间相互独立

  2. 一个物品不会被放在两个收纳箱里

如果我们将物品和收纳箱对应到建模设计中,收纳箱里装物品,而软件是领域模型负责需求点的满足,则可以这样对应:

  1. 物品对应需求点

  2. 收纳箱对应领域模型

基于这样的对应,我们可以推导出我们在需求分析和建模时候,哪些部分是主观的,哪些部分又是客观的。

主观的部分:

  1. 有多少个领域模型

  2. 哪些需求点由哪个领域模型来满足

客观的部分:

  1. 领域模型之间相互独立

  2. 一个需求点不被两个领域模型来满足

回到领域驱动设计

如果你是跟着我的思路读到这里,并且认同前面的推导逻辑以及系列前文的核心观点“领域驱动是一种价值观”,这种价值观是“边界明确是最重要的事”,最精彩的部分来了,你会发现,我们前面推导出来客观的部分,是完全契合领域驱动设计价值观的。而主观的部分,甚至都不曾在领域驱动设计概念里提及。

所以,回到文章的开头关于凭经验的说法,我们认为软件设计中,凭经验的部分不会影响领域驱动设计价值观的践行。

那么我们的核心观点是,一个决策是否符合领域驱动设计,是可以客观判断的,领域驱动设计客观上可以落地执行,并不依赖人的经验

主观的部分不重要吗

我们再把建模时候主观的部分列出来:

  1. 有多少个领域模型

  2. 哪些需求点由哪个领域模型来满足

这部分当然也很重要,经验越丰富越容易更快速地定义出职责分配更准确的模型,但它不如“边界明确”这件事重要,因为“边界明确”解决的是结构性的问题,而主观的决策部分解决的是“局部准确性问题”,这就好比建筑里的“承重墙”和“隔间墙”之间的关系,显然承重墙决定了整体的架构,对建筑最终效果以及未来改造成本有更深远的影响。

后续

到此,《老肖的领域驱动设计之路》系列的整体逻辑已经接近闭环,下一期将从“反DDD”的视角来剖析软件行业的乱象,以及开发者可以努力的方向。

主观与客观,破除DDD凭经验魔咒的更多相关文章

  1. 在单体应用的一些DDD实践经验

    阅读此文需要一定的DDD基础,如果你是第一次接触DDD读者,建议先去阅读一些DDD相关的书籍或者文章之后再来阅读本文. 背景 自从我在团队中推行DDD以来,我们团队经历了一系列的磨难--先是把核心项目 ...

  2. C轮魔咒:智能硬件为什么融资难

    据相关媒体不完全统计,2015年完成融资的智能硬件公司集中在A轮和B轮,但能挺进C轮的少之又少.对智能硬件创业的年终盘点也显示,此前比较热门的手环.智能家居等主要单品在去年明显出现了回落.陷入C轮魔咒 ...

  3. Atitit.研发管理如何避免公司破产倒闭的业务魔咒

    Atitit.如何避免公司破产倒闭的业务魔咒 1. 大型公司的衰落或者倒闭破产案例1 1.1. 摩托罗拉1 1.2. 诺基亚2 1.3. sun2 2. 为什么他们会倒闭?? 常见的一些倒闭元素2 2 ...

  4. BZOJ4516: [Sdoi2016]生成魔咒 后缀自动机

    #include<iostream> #include<cstdio> #include<cstring> #include<queue> #inclu ...

  5. C++之路进阶——HDU1880(魔咒词典)

    ---恢复内容开始--- New~ 欢迎参加2016多校联合训练的同学们~ 魔咒词典 Time Limit: 8000/5000 MS (Java/Others)    Memory Limit: 3 ...

  6. HDU 1880 魔咒词典(字符串哈希)

    题目链接 Problem Description 哈利波特在魔法学校的必修课之一就是学习魔咒.据说魔法世界有100000种不同的魔咒,哈利很难全部记住,但是为了对抗强敌,他必须在危急时刻能够调用任何一 ...

  7. Atitit.如何避免公司破产倒闭的业务魔咒

    Atitit.如何避免公司破产倒闭的业务魔咒 1. 大型公司的衰落或者倒闭破产案例1 1.1. 摩托罗拉1 1.2. 诺基亚2 1.3. sun2 2. 为什么他们会倒闭?? 常见的一些倒闭元素2 2 ...

  8. hdu 1880 魔咒词典

    题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=1880 魔咒词典 Description 哈利波特在魔法学校的必修课之一就是学习魔咒.据说魔法世界有10 ...

  9. The Lisp Curse /Lisp魔咒

    The Lisp Curse /Lisp魔咒 http://winestockwebdesign.com/Essays/Lisp_Curse.html 英文出处 http://www.soimort. ...

  10. BZOJ 4516: [Sdoi2016]生成魔咒 [后缀自动机]

    4516: [Sdoi2016]生成魔咒 题意:询问一个字符串每个前缀有多少不同的子串 做了一下SDOI2016R1D2,题好水啊随便AK 强行开map上SAM 每个状态的贡献就是\(Max(s)-M ...

随机推荐

  1. Nuxt框架中内置组件详解及使用指南(四)

    title: Nuxt框架中内置组件详解及使用指南(四) date: 2024/7/9 updated: 2024/7/9 author: cmdragon excerpt: 摘要:本文详细介绍了Nu ...

  2. 数据存储为json或数据库

    1.数据存储 In [ ]: import requests from bs4 import BeautifulSoup rqq = requests.get('http://www.tipdm.co ...

  3. 解决方案 | vba批量冻结首行,所有sheet一次性设置

    Sub FreezeTopRowAllSheets() Dim ws As Worksheet ' 遍历所有工作表 For Each ws In ThisWorkbook.Worksheets ' 激 ...

  4. 解决方案 | AutoCAD 版本+版本号+受支持的 .NET SDK版本+.NET Framework版本

    关于 Managed .NET 兼容性 Managed .NET 应用程序通常与扩展基于 AutoCAD 的产品的行为和功能的公司和第三方应用程序关联. 在移植到最新版本后,并非所有 .NET 应用程 ...

  5. 如何查看Chrome内核版本

    Blink Google chrome即谷歌浏览器原来采用的渲染引擎是Webkit,自chrome 28开始,谷歌浏览器放弃了Webkit,改用自主开发的渲染引擎Blink. 所以现在大多数喜欢尝鲜的 ...

  6. 引入样式在Element UI (Vue 2)和Element Plus (Vue 3)中的不同

    引入样式 Element UI (Vue 2):   import 'element-ui/lib/theme-chalk/index.css'; Element Plus (Vue 3):   im ...

  7. Swift开发基础08-高阶函数

    高阶函数是指接受其它函数作为参数,或者返回其它函数的函数.Swift 提供了许多内置的高阶函数,这些函数在处理集合类型数据(如数组.集合等)时尤其有用.常见的高阶函数包括 map.filter.red ...

  8. vue小知识~注入provide!

    注入表示的是将该组件的相关值,方法,实例向后代组件注入. 祖先元素中定义注入: export default { provide() { return { provideName: provideVa ...

  9. GeoScene Enterprise 3.1 临时许可更新

    Portal许可更新 portal 的许可更新很简单,直接打开Portal在线更新就好了 平台管理 -> 许可管理 -> 附加许可 -> 导入许可 -> 选择文件(选择授权的j ...

  10. 【JavaScript】从N个下拉动态监听改变的option值

    同事因为这个问题人傻了,是从Ajax请求获取的动态数据遍历的表格 然后表格行的单元格又有下拉选择,有N个下拉,要取出选择的值进行二次请求 <select name="A" i ...