.NET领域驱动设计—看DDD是如何运用设计模式颠覆传统架构

简介:

阅读目录:

  • 1.开篇介绍

  • 2.简单了解缘由(本文的前期事宜)

  • 3.DomainModel扩展性(运用设计模式设计模型变化点)

    • 3.1.模型扩展性

    • 3.2.设计模式的使用(苦心专研的设计模式、设计思想可以随意使用了)

    • 3.3.部分类的使用(封装内部对象)

    • 3.4.高强度的OO设计(面向特定领域的高度抽象设计形成特定领域框架)


  • 4.DomainModel业务逻辑规则配置(将扩展点分离后使用适当的配置将规则IOC进去)

  • 5.DDD简单总结(DDD是什么?它是“战术”)


1】开篇介绍

这篇文章不会太长,但是绝对让你对DDD有一个比较直观的认可;

这篇文章所讲到的内容虽然不多但是不太容易被领悟(因为多数人对DDD的理解还是存在很大误区的;),当然也不是多么神奇的东西,只不过是本人最近一直研究DDD的成果一个小小的心得与大家分享一下;本文讲的这些设计方式本身就存在着很大优势,你会发现它与传统三层架构最明显的区别,这也是最有经典优势的地方,最有价值的地方;

本来这篇文章是“[置顶].NET领域驱动设计—实践(穿过迷雾走向光明)”一文的一部分但是由于时间关系,完整的示例并没有跟文章同步发布,说实话时间太紧,写示例的目的是想全面的且细致的阐述DDD的分析、设计各个环节的最佳实践;原本想将文章的示例做好后在发布,但是由于工作关系和一些私人原因可能有一段时间不更新博客,又不想这篇文章拖的太久,所以我总结了两点比较有价值的地方分享给大家,目的不是让大家能会使用DDD来设计系统,而是能有一个突破点来衡量DDD到底比传统三层好在哪里,因为大部分人还没有DDD的开发经验所以能体会到应该没有相关途径;

网上很多的DDD文章有的还很不错,但是本人也是从对DDD一窍不通再到目前对DDD有一个整体的了解,觉得最大的问题是让能没有接触DDD的朋友能最贴切的体会到DDD到底哪里好,而不是一上来就大片的理论还一些UML模型图;其实完整的示例也只有这两点最有价值了,因为它是DDD所强调的中心;

2】.简单了解缘由(本文的前期事宜)

开始本文下面的章节之前先来了解一下我们将要做什么设计,我假设您没有时间阅读“[置顶].NET领域驱动设计—实践(穿过迷雾走向光明)”一文,比较文章也有点长了,所以这里简单介绍一下连续性的内容;

这篇文章我们将运用两个常规的框架设计方法来对核心的业务进行细粒度的分解设计,在以往这点很难实现,所以我为什么要说框架的设计思想,因为我们对设计模式的运用主要在框架、组件这些非业务需求性的基础设施上;那么这里我们将用这些强大的武器来对最难对付的业务扩展性的设计;

本文全部的业务其实是一个简单的学习考试系统的背景,我们下面将要运用强大的设计能力来对【Employee】聚合进行细粒度的设计、配置;之前的设计已经全部结束,数据持久化也设计完成,就剩下编码阶段;编码的最大挑战就在于前期的相关接口的设计,这里是细粒度的接口设计,不是简单的分分层;

图1:

163443363.jpg

(查看大图)

上图中我用红圈标记出我们下面要扩展的【Employee】聚合,在将模型落实到代码后我们将要通过规约模式来将【Employee】的验证对象化,然后通过设计模式的策略模式将规则策略化,再通过Configuraion Manager来管理所有的业务规则的配置,这个时候IOC就派上用场了,一切都很顺手;传统三层你是无法做到的;

请看下面【Employee】实体类代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
/*==============================================================================
  * Author:深度训练
  * Create time: 2013-07-08
  * Blog Address:http://www.cnblogs.com/wangiqngpei557/
  * Author Description:特定领域软件工程实践;
  * ==============================================================================*/
namespace  Domain.DomainModel.ExaminationModule.Aggregates.EmployeeAgg
{
     using  System;
     using  System.Collections.Generic;
     using  Domain.DomainModel.ApproveModule.Aggregates.ParentMesAgg;
     using  Domain.DomainModel.ExaminationModule.Aggregates.FieldExaminationAgg;
     using  Domain.Seedwork;
     public  partial  class  Employee : EntityObject
     {
         public  Employee()
         {
             this .ParentMessage =  new  HashSet<ParentMessage>();
             this .TeacherCheckGroup =  new  HashSet<TeacherCheckGroup>();
         }
         public  string  EID {  get set ; }
         public  string  EName {  get set ; }
         public  Nullable<Employee_Marry> IsMarry {  get set ; }
         public  Nullable< int > Age {  get set ; }
         public  string  UserName {  get set ; }
         public  string  PassWord {  get set ; }
         public  Nullable<Employee_Switch> SWitch {  get set ; }
         public  Nullable<Employee_Role> EmpRole {  get set ; }
         public  Nullable<Employee_Sex> Sex {  get set ; }
         public  virtual  ICollection<ParentMessage> ParentMessage {  get set ; }
         public  virtual  ICollection<TeacherCheckGroup> TeacherCheckGroup {  get set ; }
         public  void  ReSwitch()
         {
             if  ( this .SWitch.Value == Employee_Switch.IsFalse)
                 this .SWitch = Employee_Switch.IsTure;
             else
                 this .SWitch = Employee_Switch.IsFalse;
         }
         public  void  Reinitial()
         {
             PassWord =  "000000" ;
         }
     }
}


【Employee】聚合跟一般的聚合没多大区别,比较简单的结构,为了看起来完整一点,我加入了两个初始化的行为;ReSwitch是用来启用、关闭当前账户;

Reinitial是初始化当前【Employee】的初始默认密码,完全是演示而用;

那么我们下面要做什么呢?在以【Employee】为聚合根里面我们聚合了【ParentMessage】家长留言、【TeacherCheckGroup】站考,两个集合,其实这是用来做导航属性的;实体框架需要这些信息做实体导航使用,在设计的时候你需要权衡你需要多少这样的关联;

现在经过我们对需求的深入分析之后可能会存在这样的变动情况:

【Parent家长】向【Employee教师】【留言】后,教师需要对留言内容做出反馈,比如要【及时的回复】,对于不同的【留言级别】需要给出不同的处理;

这个需求很简单,但是它里面透露出来的是什么?设计的扩展性,这个扩展性在哪里?对于不同的【留言级别】需要给出不同的【处理】,很显然是一个可能随时会变化的点;

【Employee_Priority】代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
/*==============================================================================
  * Author:深度训练
  * Create time: 2013-07-08
  * Blog Address:http://www.cnblogs.com/wangiqngpei557/
  * Author Description:特定领域软件工程实践;
  * ==============================================================================*/
namespace  Domain.DomainModel.ApproveModule.Aggregates.ParentMesAgg
{
     using  Domain.DomainModel.ExaminationModule.Aggregates.EmployeeAgg;
     using  System;
     public  partial  class  ParentMessage
     {
         public  string  PMID {  get set ; }
         public  string  PID {  get set ; }
         public  string  EID {  get set ; }
         public  string  Content {  get set ; }
         public  Nullable<Message_Priority> Priority {  get set ; }
         public  Nullable<System.DateTime> Datetime {  get set ; }
         public  virtual  Employee Employee {  get set ; }
         public  virtual  Parent Parent {  get set ; }
     }
}

有一个Priority属性,是标记该留言的紧急情况,看代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/*==============================================================================
  * Author:深度训练
  * Create time: 2013-07-08
  * Blog Address:http://www.cnblogs.com/wangiqngpei557/
  * Author Description:特定领域软件工程实践;
  * ==============================================================================*/
namespace  Domain.DomainModel.ExaminationModule.Aggregates.EmployeeAgg
{
     using  System;
     public  enum  Message_Priority :  int
     {
         Normal = 1,
         Pressing = 2
     }
}

有两种级别,Normal表示普通的,Pressing表示紧急的,对于紧急的肯定是需要先处理的,而且处理的逻辑或多或少有点不同;在DDD中所有的业务逻辑都要在DomainModel Layer 中处理,这是原则;所有的逻辑都不是直接使用,比如在登录的时候我们通常是验证用户名密码是否真确,但是通常还会有很多其他的条件,比如说当前用户是否是高级会员、是否欠费等等,这些都是在聚合规约工厂中统一获取的,这就便于我们将变化的点抽到专门的地方进行设计;

逻辑判断的地方原则是不直接写IF\ELSE,逻辑处理地方原则是不直接写实现代码,通过接口实现策略类;

图2:

163806313.jpg

我们在【Employee】中加入了一个对【ParentMessage】实体的处理;由于我们的DomainModel通常不是直接持久化在MemberCache中的,所以对于有UI交互的操作都无法很好的进行实体直接使用,如果是自动化的操作这里的方法就不需要任何参数,每次都需要将留言的ID带过来,然后我们再进行内部的查找;当然这里还可以在Application Layer 就把【ParentMessage】实例拿到穿进来也可以;

其实这个时候已经开始将进行细粒度的设计了,我们看一下DomainModel结构;

图3:

163841244.jpg

如果是数据库驱动,我们是无法提取出【Employee】的相关对象的,一些状态也只是数字表示而已缺乏OO思想,也就谈不上面向对象的设计了;这里最让人欣喜诺狂的是我们已经完全可以将【Employee】相关的逻辑进行细粒度的扩展性设计了,这里我们将把所有跟【Employee】相关的所有业务逻辑都放入专门EmployeeLogic目录中;这样的好处真的很多,跟我们最相关的就是项目的任务分配,这样设计完成后就完全可以将某些逻辑抽象出来接口分配给某人去实现;

这一节主要就是介绍一下相关的背景,下面我们就要将对【Employee】处理【ParentMesssage】的业务逻辑进行高度的分析、设计、配置化;

3】DomainModel扩展性(运用设计模式设计模型变化点)

模型扩展性是一个一直被我们关注无数次提起的焦点,对它的把握始终未能实现;传统分层架构将所有的业务逻辑洒满每个层面,从UI层到数据库都有多多少少的业务逻辑,而不是各负其责,管好自己分类的事情;UI层主要负责自己的样子好看,不要这里弄脏了那里弄脏了;数据库应该管好数据的存储,数据的优化等等,两者都没有权利去管业务逻辑的权利;

这里我们将要通过设计模式将对可能存在变化的业务逻辑抽象出来进行设计;

3.1】模型扩展性

在上面的介绍总我们大概了解了需求,下面我们要通过对【ParentMessage】的Priority属性进行判断,因为这两种优先级对于业务逻辑处理是不同的,但是可能会存在着相同的逻辑,这就完全符合我们的OOA、OOP的中心了,我们可以进行强大的抽象继承来处理,这也是我们OO应该管理的范围,UI\数据库都不具备这样的能力;

可以将DDD与UI、数据库打个比方:

UI:我没有什么事情,分点业务给我处理吧;

数据库:我很强大,所有的数据都在我的管理范围之内,我想怎么处理就怎么处理,我天下第一;

DDD说:各位兄弟,要么从一开始的时候就听我的,要不然后面出了什么事,我管不了你们了;——王清培;

设计模式很强大,能处理当前业务问题的有很多模式可以选择,这里我们使用常用的“策略模式”来解决不同Priority的逻辑;

3.2】设计模式的使用(苦心专研的设计模式、设计思想可以随意使用了)

设计模式的强大不需要我再来废话了,大家都懂;那么这里我们需要将逻辑的处理抽出来放入专门的逻辑处理类中去,这也符合向扩展开放向修改封闭原则;

将逻辑的处理独立出去,跟DomainModel之间存在着一个带有阴影的重贴关系,虽然逻辑处理类相对独立当时它毕竟还是处于领域类的东西;将业务逻辑完全的封闭在领域层中,但是在这个层中不是胡子眉毛一把抓,还是需要就具体的业务进行细粒度的分析、设计,对架构师来说是一个不小的挑战,因为大部分的架构师比较关注纯技术的东西,对业务的分析设计缺乏经验和兴趣;

我们来看一下对Priority的处理简单设计:

图4:

163909980.jpg

最理想的设计就是面向接口,【Employee】实体不会依赖哪一个具体的实现类;

图5:

163933124.jpg

我们对Priority的处理逻辑抽象出来了相关的策略接口IParentMessageOperation,该接口有两个接口分别用来处理不同优先级的具体业务逻辑;ParentMessageOperationNormal是处理Priority为Normal的逻辑,ParentMessageOperationPressing是处理Priority为Pressing的逻辑,当然为了后面考虑我又加了一个abstract calss ParentMessageOperationBasic 做一些相同逻辑的抽象;

一个简单的Priority的逻辑都可以这样去设计,我想再复杂的业务只要业务分析好,这里的设计是不会有问题;到目前为止我都在为DDD的强大敢到震惊,我不相信你没有看出来它能把多么复杂的问题简单化,以往是绝对不可能完成这样的设计的,至少我从来没看见过也没听过谁能在传统三层架构下把复杂的业务系统设计的很灵活,而且又不会污染UI、数据库;

有策略接口那么我们还得把相应的实现类给绑上去,这里有两种方式,第一种使用简单接口直接判断然后创建策略实现,第二种是使用IOC的方式动态的注入进来,当然这里已经到了我们大家都比较擅长的范围了,每个人的设计思想不同就不多废话了;

图6:

164001673.jpg

看着这样的结构,我没有理解再说DDD不优雅;到了这里已经很清晰了,我们使用IParentMessageOperationFactory创建IParentMessageOperation,具体的逻辑封装在以IParentMessageOperation接口为主的实现类中;

图7:

164028660.jpg

我们通过IParentMessageOperationFactory创建IParentMessageOperation实现,是不是很清爽;

图8:

164057555.jpg

这里的代码几乎是不会随着业务的变化而变化,要变化的是在逻辑处理里面;

图9:

164132272.jpg

接口的处理逻辑方法,很简单约定一个【ParentMessage】、【Employee】两个实体,这里需要注意平衡实体之间的关联性;

图10:

164202292.jpg

通过基类可以抽象点公共的逻辑,这里是为了演示而用;其实到了这一步大家都知道怎么来进行设计了,关键是要分析好业务,然后得出深层领域模型,在此基础上进行设计才是靠谱的,不要为了设计而设计,不要陷入技术的困境;

图11:

164240957.jpg

该图是我们对priority相关逻辑的设计,顶层是两个接口,右边是一个Factory实现,左边是Operation的继承树,还是比较简单的;

3.3】部分类的使用(封装内部对象)

在很多时候我们的设计需要借助部分类来规划对象的关系,以免污染其他的实体;比如这里的【Employee】需要在内部使用一个特定的类型,那么最好是放在【Employee】内部使用,不要暴露在外面;这点在逻辑处理中进行设计比较合理;

图12:

164310746.jpg

内部类再配合泛型一起用将发挥很大的设计奇效,这里就不扯了;

3.4】高强度的OO设计(面向特定领域的高度抽象设计形成特定领域框架)

从上面的3.3】节中我们能体会到,对于特定领域的抽象其实是可行的,也就是说最终会形成强大的面向特定领域的框架、组件,但是这样的框架是不通用的,也就是当前领域模型才能使用,这对于一般的项目而言确实成本很大,得不偿失;然后对于需要长期维护的项目、产品、电子商务平台值得投入,长期重构得出内聚性很强的领域框架;

图13:

164357478.jpg

如果你一个框架做通用性的功能,只能做到泛泛而已,无法深入到业务内部;

图14:

164441621.jpg

其实就是将精力集中在特定领域而已,逐渐重构出特定领域的框架;

4】DomainModel业务逻辑规则配置(将扩展点分离后使用适当的配置将规则IOC进来)

其实到了这里,再说将业务逻辑配置化已经不是什么大问题了,只需要将上面的IParentMessageOperation实现类通过IOC的方式配置好;但是这个配置的策略需要结合业务来判断,可能存在多维度的判断才能最终确定使用哪一个实现类,扯远点如果后面配合C#4.0的元编程其实真的可以实现运行时配置逻辑策略了,但是目前来看不是很成熟;我们只有先将所有的业务逻辑实现好,然后根据业务需要进行后台配置;

比如系统的后台管理自动检测是否是休息天,如果是休息天那么对于【Employee】就没有权利去执行【ParentMessage】的处理,是不是很简单?当然很多好的设计可以慢慢的搬到系统中来,前提是“特定领域重构—特定领域框架设计”,这个度好把握好;

5】DDD简单总结(DDD是什么?它是“战术”)

最近园子里讨论.NET技术值钱不值钱的文章很火,其实不管是.NET还是\JAVA都是工具,战斗的工具,必须具备一定的战略设计才能让他们彼此发挥具体的作用;可以把DDD比喻成孙子兵法,.NET只是打仗时的一个工具,JAVA也是如此,Python、ruby等等,关键是设计思想、战略;所以我们长期培养的是设计能力,适当的熟悉某一种技术平台,以不变应万变;JAVA在牛逼,不懂企业架构一样是垃圾,.NET再牛逼,不懂设计模式一样玩不转;

所有的技术框架都有其优缺点,我们只有先进行总体的设计、规划,然后在适当的位置使用适当的技术,这个技术在这个方面比较擅长,那么就把它安排在这个位置;.NET优势在开发速度、UI上,那么就用来进行前台部分的开发;JAVA可能在大数据、分布式后端有优势,那么用来做服务器开发,搜索引擎;Ruby是动态语言,可以用来实现复杂的业务动态配置,集众家之所长来完成一次大型的战役,没有谁离开谁转不了,没有谁比谁更重要,在一次战斗中连火头军都是不能少的,杨门女将中的杨排风谁看小看它,唯有军师不能糊涂;谢谢;


示例DEMO代码(领域模型):http://files.cnblogs.com/wangiqngpei557/Domain.DomainModel.zip





 本文转自 王清培 51CTO博客,原文链接:http://blog.51cto.com/wangqingpei557/1275769,如需转载请自行联系原作者



相关文章
|
10天前
|
设计模式 存储 前端开发
揭秘.NET架构设计模式:如何构建坚不可摧的系统?掌握这些,让你的项目无懈可击!
【8月更文挑战第28天】在软件开发中,设计模式是解决常见问题的经典方案,助力构建可维护、可扩展的系统。本文探讨了.NET中三种关键架构设计模式:MVC、依赖注入与仓储模式,并提供了示例代码。MVC通过模型、视图和控制器分离关注点;依赖注入则通过外部管理组件依赖提升复用性和可测性;仓储模式则统一数据访问接口,分离数据逻辑与业务逻辑。掌握这些模式有助于开发者优化系统架构,提升软件质量。
27 5
|
10天前
|
XML 开发框架 .NET
.NET框架:软件开发领域的瑞士军刀,如何让初学者变身代码艺术家——从基础架构到独特优势,一篇不可错过的深度解读。
【8月更文挑战第28天】.NET框架是由微软推出的统一开发平台,支持多种编程语言,简化应用程序的开发与部署。其核心组件包括公共语言运行库(CLR)和类库(FCL)。CLR负责内存管理、线程管理和异常处理等任务,确保代码稳定运行;FCL则提供了丰富的类和接口,涵盖网络、数据访问、安全性等多个领域,提高开发效率。此外,.NET框架还支持跨语言互操作,允许开发者使用C#、VB.NET等语言编写代码并无缝集成。这一框架凭借其强大的功能和广泛的社区支持,已成为软件开发领域的重要工具,适合初学者深入学习以奠定职业生涯基础。
67 1
|
16天前
|
设计模式 存储 前端开发
MVC革命:如何用一个设计模式重塑你的应用架构,让代码重构变得戏剧性地简单!
【8月更文挑战第22天】自定义MVC(Model-View-Controller)设计模式将应用分为模型、视图和控制器三个核心组件,实现关注点分离,提升代码可维护性和扩展性。模型管理数据和业务逻辑,视图负责数据显示与用户交互,控制器处理用户输入并协调模型与视图。通过示例代码展示了基本的MVC框架实现,可根据需求扩展定制。MVC模式灵活性强,支持单元测试与多人协作,但需注意避免控制器过度复杂化。
25 1
|
25天前
|
存储 消息中间件 JSON
|
9天前
|
消息中间件 Java RocketMQ
微服务架构师的福音:深度解析Spring Cloud RocketMQ,打造高可靠消息驱动系统的不二之选!
【8月更文挑战第29天】Spring Cloud RocketMQ结合了Spring Cloud生态与RocketMQ消息中间件的优势,简化了RocketMQ在微服务中的集成,使开发者能更专注业务逻辑。通过配置依赖和连接信息,可轻松搭建消息生产和消费流程,支持消息过滤、转换及分布式事务等功能,确保微服务间解耦的同时,提升了系统的稳定性和效率。掌握其应用,有助于构建复杂分布式系统。
26 0
|
10天前
|
机器学习/深度学习 并行计算 算法
深度学习驱动的声音生成:FunAudioLLM的创新架构
【8月更文第28天】随着深度学习技术的发展,声音合成的质量得到了显著提升。本文将介绍 FunAudioLLM —— 一种基于深度学习的声音生成框架,旨在创造高质量、自然流畅的声音内容。我们将探讨 FunAudioLLM 的核心技术、训练流程及其实现细节,并提供一些示例代码。
11 0
|
1月前
|
缓存 前端开发 项目管理
业务驱动的应用架构设计
业务驱动的应用架构设计
30 1
|
15天前
|
BI
软件设计与架构复杂度问题之业务简单的系统不适合使用DDD架构如何解决
软件设计与架构复杂度问题之业务简单的系统不适合使用DDD架构如何解决
|
2月前
|
设计模式 存储 运维
微服务架构中的服务发现与注册中心设计模式
在现代软件工程实践中,微服务架构已成为构建灵活、可扩展系统的首选方案。本文将深入探讨微服务架构中至关重要的服务发现与注册中心设计模式。我们将从服务发现的基本原理出发,逐步解析注册中心的工作机制,并以Eureka和Consul为例,对比分析不同实现的优劣。文章旨在为开发者提供一套清晰的指导原则,帮助他们在构建和维护微服务系统时做出更明智的技术选择。
|
1月前
|
存储 安全 数据管理
业务驱动的数据架构设计
业务驱动的数据架构设计
27 0
下一篇
DDNS