领域驱动设计,让程序员心中有码(五)

1      从搬砖谈领域对象

  有一个古老的故事,大概是这样的。作者问三个建筑工地上的工人他们在干什么?有一个没精打采的说,我在挖洞!而另一一个人却说,我在盖一座房子。还有一个人说,我在建立一座巨大的城市。不同的思维模式决定了不同的发展,十年过后,第一个工人,还是在挖洞,而第二个则成为了工头。第三个最终却成为了大设计师。

  在软件开发领域,往往会使用搬砖这个词来形容我们所开发的每个功能模块,实际上也确实如此,如果把我们需要完成的每个项目,比作一座高楼大厦,那么在项目中所完成的各种模块,也确实是我们在计算机世界中利用砖块设计出来的精美建筑构建。而从领域驱动的角度来说,可以把关系,类比为建筑工程图纸中使用的各种辅助线,也可以把领域驱动中所涉及的各个对象,类比成砖块,这些砖块,大概有两种:一种是实体(Entity),一种是值对象(Value Object),而使用这些对象的工具,则成为服务(Service),完成的各个建筑构建,被成为包或者模块(Module).

2      关联关系

  在介绍领域驱动设计的第三篇文章《领域驱动设计,让程序员心中有码(三)》中,笔者提到了UML中常用的几种关系,而关联关系是一种最为常见的关系。在软件设计过程中,无所不在的关联,有时候会让软件工程设计变得更加复杂。因此,在设计关联关系时,应该让关联更加易于控制,这意味着需要采取下列三种措施:

  1、规定一个遍历方向。对象与对象间,过于双向关联是一种低效的关系,而指定唯一的遍历方向,将有效的减少相互的依赖,实现设计的简化。

  2、添加一个限定符,以便有效地减少多重关联。过于复杂的多对多关系,最终形成一个纷繁复杂难以控制的图结构,而限定多对多关联的遍历方向,可以有效的简化多对多关系为一对多关联。

  3、消除不必要的关联。上述两个步骤的目的,也正是为了消除对于当前工作或模型对象的基本含义来说不重要的关联。实际上正是为了当前模型对象的简化。

3      实体

  在软件开发过程中,我们通常会定义模型和实体对象,这种实体对象同样也是领域驱动中的基本对象。按照大家的理解,通常而言,实体是指能够与数据库直接映射的对象。在领域驱动设计中,使用的则是更加妥当的说法:对象具有贯穿整个生命周期(甚至会经历多种形式)的抽象的连续性。 实体标识任何事物,只要满足两个条件即可:一个是它在整个生命周期中,具有联系性,二是他的区别并不是有哪些对用户来说非常重要的属性决定,而是通过标识来决定的。

    3.1   实体建模

  由于实体对象的基本职责是为了确保连续性,其行为应该是非常清楚并且可以预测的。因此保持实体的简练是实现责任的关键。应该抓住实体的基本特征,而不要一味地过分求全求完美。对于实体而言,应该只添加对概念来说至关重要的行为和这些行为所必须的属性。其他行为,应当转移到与核心实体关联的其他对象中。实体则通过协调与之关联的其他对象来完成自己的基本职责。

3.2   设计实体的标识

  在面向对象开发中,会使用建立标识这种操作方式来实现与其他对象的区分。哪怕是在分布式系统中,同样需要使用标识来确保标识的唯一性。可以使用具有唯一性的属性来提供标识,也可以使用ID的方式来实现。这种ID如果使用系统自动生成,往往需要有一些手段确保生成的唯一性,尤其是在分布式系统中,更是一个非常困难的问题。经常使用的方式是使用redis或zookeeper这些中间件来生成唯一标识,还有一种常见的方案是使用twitter的Snowflake算法,这些算法就不再赘述了。

4      值对象

  值对象则不具备Entity这种明确的连续性,如果在设计系统时,将所有的对象都定义为实体对象,实际上将会极大的增加系统的复杂度,所以需要定义一些用于描述领域的某个方面,本身没有概念标识的喜爱那个。例如,可以通过邮编对地址进行检索,邮编的变更,对地址也可能会发生变化,那么地址就是具有连续性的实体对象。而在电子商务系统中,只需根据地址即可完成投递,而无需确保地址的连续性,那么他就是值对象了。

  值对象,往往使用与需要通过一个模型元素的属性来定义模型的场景,主要作为参数在对象间传递消息。通常是临时对象,在操作结束后,就可以被丢弃。值对象可以作为实体的属性,例如,一个人,是一个完整的实体,而他的名字,则是值对象。当然,也并非意味着值对象是一个单纯的属性,实际上值对象是指某一个特定概念下,具有完整意义的、通过属性进行理解的对象。例如,地址由省、市、区、街道、邮编等综合属性组成,这些组成对象,实际上也是实体,他们联系起来,就组成了值对象。      

5      服务

  在软件设计中,并非所有的对象都需要通过标识或属性进行区分。领域驱动设计中,使用服务(Service)来定义具有活动或动作的对象。事实上也确实如此,并非所有的对象都适合使用实体或值对象来进行建模。服务强调与其他对象的操作,是通过定义能够为使用者做什么来实现的。也就是说,服务倾向于动词领域,而非名词领域。

  5.1   服务对象的基本特征

  按照领域驱动设计的说法,一个好的服务应该具有以下特征:

  1)与领域概念相关的操作,不是Entity或ValueObject的一个自然组成部分。

  2)接口是根据领域模型的其他元素定义。

  3)操作是无状态的。操作的无状态是指任何调用者都能使用,而无需关注实例的历史状态。

5.2   服务与领域层

  在领域涉及中,服务无处不在,大体上包括以下几种不同层次。

  1、应用层:定义与应用相关的基础服务,例如在处理资金转账业务时,定义一系列服务,1、包括获取输入,2、发送消息给领域层服务,由其完成动作的执行;3、监听确认消息等。

  2、领域层:处理与相关的服务,例如,处理有上述转账业务发起的请求,例如进行结果的确认等。

  3、基础设施层:发送消息通知。

  5.3   服务的粒度

       在概念建模中,通过控制领域层中接口的力度,可以有效的实现客户端与实体和值对象的耦合。通过合理的模式确保接口的简单性,将便于在大型或分布式系统中对组件进行打包的粒度控制,这实际上也是微服务架构中,服务粒度细分的理论基础。

6      包或模块

  模块,是软件工程学中自古有之的基本概念。在软件系统设计中,经常会按照各种各样的类别进行分解,有时候按照技术架构来分割,有时候则按照开发者的任务例如按照用例来进行细分,有的在软件重构过程中,甚至会沿用历史架构早期形成的模块划分。

  在软件工程学中,高内聚,低耦合是基本的概念,而在模块之间的关系,成为耦合,而模块内部的关系,成为内聚。因此,好的软件项目,模块之间应该低耦合,而模块内部则应该高内聚。但是模块的划分,跟软件分层划分一样,不应该仅仅只是代码层面的划分,而应该是概念模型角度的划分。不连贯的思想或者“一锅粥”式的模块划分,最终只会造成系统开发的严重不可控。

  领域驱动设计认为,模块,是一种非常重要的表达机制。模块的选择应该取决于被花费到模块中的对象的意义。当某些对象在模块中被创建时,实际上相当于告诉下一位开发者,这些对象间是通过模块来实现了某种关系。

  选择能够描述系统的模块,并使之饱含一个内聚的概念集合。应该基于模块来实现概念组合的方式,从而可以向相互独立地理解和分析这些概念。对模型进行精化,直到可以更具高层领域概念对模型进行划分,同时,相应的代码也不会产生耦合。

7      结论

  随着系统设计规模和复杂度的增加,模块化变得更加重要。领域模型中的每个概念都需要在实现元素中反映出来。实体、值对象、他们之间的关联关系、领域服务以及用于组织元素的模块都是实现领域模型相对应的地方。实现中的对象、指针和检索机制必须直接、清楚地映射到模型对象。

相关文章:

  • 领域驱动设计,让程序员心中有码

  • 领域驱动设计,让程序员心中有码(二)

  • 领域驱动设计,让程序员心中有码(三)

  • 领域驱动设计,让程序员心中有码(四)

原文地址: https://www.cnblogs.com/xiyuanMore/p/10230801.html

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

640?wx_fmt=jpeg


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/317949.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

.NET Core实战项目之CMS 第十四章 开发篇-防止跨站请求伪造(XSRF/CSRF)攻击处理...

通过 ASP.NET Core,开发者可轻松配置和管理其应用的安全性。 ASP.NET Core 中包含管理身份验证、授权、数据保护、SSL 强制、应用机密、请求防伪保护及 CORS 管理等等安全方面的处理。 通过这些安全功能,可以生成安全可靠的 ASP.NET Core 应用。而我们这…

模板:左偏树

文章目录解析可以解决的问题定义:左偏树的基本性质基本结论操作合并访问与删除堆顶元素插入元素批量插入删除已知元素所谓左偏树,就是往左偏的树 下面介绍一下它的一个兄弟: 《右偏树》 (逃) 解析 所谓左偏树&#…

迎开学水题狂欢赛(舞踏会[dp+三叉树],HH去散步[矩阵快速幂],排序[模拟],铁路旅行[线段树])

快速简单记录老师口胡(可能就我自己看得懂了吧…) 文章目录T1:舞踏会titlesolutioncodeT2:HH去散步titlesolutioncodeT3:排序titlesolutioncodeT4:铁路旅行titlesolutioncodeT1:舞踏会 title …

CSP2021提高组复赛解析

前言 终于出成绩了我可以写博客辣,官方数据还没出就先放洛谷的题目链接了。 正题 T1-廊桥分配 https://www.luogu.com.cn/problem/P7913 题目大意 有m1m_1m1​种一类飞机,m2m_2m2​种二类飞机,每个飞机有一个占用时间的区间。要给两类飞机…

一起开心集训队第一周训练赛2021/3/14

文章目录比赛链接A CodeForces 1481D AB Graph题意:题解:代码:B CodeForces 1481E Sorting Books题意:题解:代码:C CodeForces 1478D Nezzar and Board题意:题解:代码:D …

使用Azure DevOps持续集成GitHub项目

点击蓝字关注我微软的Azure DevOps是一款软件开发管理工具,整合了需求、代码、编译、测试、发布等所有功能于一身。今天我们就来看看如何用Azure DevOps对自己GitHub上的项目做持续集成,并能在GitHub显示最新编译状态。其实在不久之前,Azure …

[BZOJ 3811]玛里苟斯(线性基)尽量理解的题解

文章目录titlesolutioncodetitle 魔法之龙玛里苟斯最近在为加基森拍卖师的削弱而感到伤心,于是他想了一道数学题。 S 是一个可重集合,S{a1,a2,…,an}。 等概率随机取 S 的一个子集 A{ai1,…,aim}。 计算出 A 中所有元素异或和,记为 x, 求 x^…

CF464E The Classic Problem(线段树 最短路)

CF464E The Classic Problem \(\bigstar\texttt{Hint}\):发现没有什么好的突破口?为什么不想想怎样才能实现题目中 \(2^x\) 的加减法呢? 可见每次加减法,我们要做的是将添加的 \(1\) 和右边的连续的 \(1\) 合并为一整段&#xff0…

C. Longest Simple Cycle

C. Longest Simple Cycle 题意: 有n条链,第i条链上有c[i]个点,a[i]为第i条链的顶点与第i-1条链的连接点,b[i]为第i条链的最后一个点与第i-1条链的连接点。通过上面的方法连接链会产生很多的环,问这些环的最大长度。 …

【CF813F】Bipartite Checking(线段树分治+可删除并查集)

文章目录titlesolutioncodetitle You are given an undirected graph consisting of n vertices. Initially there are no edges in the graph. Also you are given q queries, each query either adds one undirected edge to the graph or removes it. After each query you…

在.Net Core WebAPI下给Swagger增加导出离线文档功能

一丶前言最近刚接触到Swagger,在github上下载了它的源码和demo学习了一遍,发现这个组件非常好用,不过不足的是它没有导出离线文档的功能,于是乎我就想给它加一个导出功能Swagger Github开源地址二丶思路其实说白了api文档就是一个…

YBTOJ洛谷P4331:数字序列(左偏树)

文章目录题目描述数据范围解析代码题目描述 数据范围 n<1e6n<1e6n<1e6 解析 先考虑简单情况 如果原数列是单调递增的&#xff0c;显然应该使biaib_ia_ibi​ai​ 如果单调递减&#xff0c;应该取中位数 那么原数列如果分成单调递减的几段&#xff0c;那么每一段都取中…

P8441 旭日东升(二维数点经典套路)

P8441 旭日东升 维护一个不可重集合的序列 \(a\)&#xff0c;长度为 \(n\)。支持以下两种操作&#xff1a; l r x 对于每个 \(l\le i\le r\)&#xff0c;将 \(x\) 并入 \(a_i\)。l r 设 \(S\) 把每个 \(l\le i\le r\) 的 \(a_i\) 并在一起的集合&#xff0c;输出 \(S\) 中所有元…

深搜、广搜、搜索剪枝

搜索与回溯讲解 文章目录深搜方向向量&#xff1a;DFS代码&#xff1a;题目讲解&#xff1a;八皇后问题字符序列自然数的拆分广搜BFS代码&#xff1a;题目讲解&#xff1a;瓷砖关系网络bfs与dfs的用途与区别搜索剪枝可行性剪枝最优性剪枝记忆化搜索搜索顺序剪枝题目&#xff1a…

使用logdashboard查看可视化日志

logdashboard日志面板是我在Github写的一个开源项目&#xff0c;旨在让查看日志变的方便快捷。在线预览现在功能有日志检索、趋势图、异常堆栈快速查看、日志详情等logdashboard支持自定义日志模型可以记录更多自定义的属性。logdashboard支持的日志来源有以下两种&#xff0c;…

数论分块专题复习(余数求和+模积和+Ice Rain+The Fool)

文章目录前提知识复习T1&#xff1a;余数求和titlesolutioncodeT2&#xff1a;Ice RaintitlesolutioncodeT3&#xff1a;The FooltitlesolutioncodeT4&#xff1a;模积和titlesolutioncode前提知识复习 整除分块是用于快速处理形似下列式子的方法&#xff0c;是解决莫比乌斯反…

领域驱动设计,让程序员心中有码(六)

领域驱动设计-聚合&#xff0c;一种极简的思维模式引言作为IT技术产业飞速发展的产物&#xff0c;软件工程学已经成为当今时代非常重要的一个学科。作为一名资深的软件开发从业者&#xff0c;我们需要学习的东西实际上已经远远超出了原本在大学教育阶段所接受的知识深度和广度&…

hdu 5094 Maze

题意&#xff1a; n*m大的迷宫 &#xff0c;有p种钥匙。钥匙最多有10种。 然后一个k&#xff0c;然后k行表示 (x1,y1),(x2,y2)直接有门或者墙。 如果g0 &#xff0c;就是有墙&#xff0c; 如果g>0 表示有门&#xff0c;且门需要第g把钥匙才能开。 然后下来一个s&#xff…

数论分块练习([CF830 C]Bamboo Partition + [hdu 6395]Sequence )

文章目录T1&#xff1a;SequencetitlesolutionT2&#xff1a;Bamboo PartitiontitlesolutioncodeT1&#xff1a;Sequence title 传送 solution 一眼就是很裸的矩阵加速 ⌊pl⌋\lfloor\frac{p}{l}\rfloor⌊lp​⌋分块矩阵加速就可以了 [BA1][DC⌊pl⌋010001]\begin{bmatrix}…

YBTOJ:染颜色(KDtree)

文章目录题目描述数据范围解析代码题目描述 数据范围 n≤105,T≤5n\le10^5,T\le5n≤105,T≤5 解析 关键是对问题的转化 考虑一次修改&#xff0c;一个点需要满足哪些条件才可以满足 1.与x的距离不超过lll 2.在x的子树内 这两个条件可以转化为对深度的限制和对dfs序的限制 这样…