0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何实现DCI架构(中)

jf_78858299 来源:元闰子的邀请 作者:元闰子 2023-05-10 17:10 次阅读

然而,充血模型并非完美,它也有很多问题,比较典型的是这两个:

问题一:上帝类

People这个实体包含了太多的职责,导致它变成了一个名副其实的上帝类。试想,这里还是裁剪了很多“人”所包含的属性和行为,如果要建模一个完整的模型,其属性和方法之多,无法想象。 上帝类违反了单一职责原则,会导致代码的可维护性变得极差

问题二:模块间耦合

SchoolCompany本应该是相互独立的,School不必关注上班与否,Company也不必关注考试与否。但是现在因为它们都依赖了People这个实体,School可以调用与Company相关的Work()OffWork()方法,反之亦然。这导致 模块间产生了不必要的耦合,违反了接口隔离原则

这些问题都是工程派不能接受的,从软件工程的角度,它们会使得代码难以维护。解决这类问题的方法,比较常见的是对实体进行拆分,比如将实体的行为建模成 领域服务 ,像这样:

type People struct {
 vo.IdentityCard
 vo.StudentCard
 vo.WorkCard
 vo.Account
}

type StudentService struct{}
func (s *StudentService) Study(p *entity.People) {
 fmt.Printf("Student %+v studying\\n", p.StudentCard)
}
func (s *StudentService) Exam(p *entity.People) {
 fmt.Printf("Student %+v examing\\n", p.StudentCard)
}

type WorkerService struct{}
func (w *WorkerService) Work(p *entity.People) {
 fmt.Printf("%+v working\\n", p.WorkCard)
 p.Account.Balance++
}
func (w *WorkerService) OffWOrk(p *entity.People) {
 fmt.Printf("%+v getting off work\\n", p.WorkCard)
}

// ...

图片

这种建模方法,解决了上述两个问题,但也变成了所谓的 贫血模型People变成了一个纯粹的数据类,没有任何业务行为。在人的心理上,这样的模型并不能在建立起对现实世界的对应关系,不容易让人理解,因此被学院派所抵制。

到目前为止,贫血模型和充血模型都有各有优缺点,工程派和学院派谁都无法说服对方。接下来,轮到本文的主角出场了。

DCI架构

DCI (Data,Context,Interactive)架构是一种面向对象的软件架构模式,在《The DCI Architecture: A New Vision of Object-Oriented Programming》一文中被首次提出。与传统的面向对象相比,DCI能更好地对数据和行为之间的关系进行建模,从而更容易被人理解。

  • Data ,也即数据/领域对象,用来描述系统“是什么”,通常采用DDD中的战术建模来识别当前模型的领域对象,等同于DDD分层架构中的领域层。
  • Context ,也即场景,可理解为是系统的Use Case,代表了系统的业务处理流程,等同于DDD分层架构中的应用层。
  • Interactive ,也即交互,是DCI相对于传统面向对象的最大发展,它认为我们应该显式地对领域对象( Object )在每个业务场景(Context)中扮演( Cast )的角色( Role )进行建模。 Role代表了领域对象在业务场景中的业务行为(“做什么”),Role之间通过交互完成完整的义务流程

这种角色扮演的模型我们并不陌生,在现实的世界里也是随处可见,比如,一个演员可以在这部电影里扮演英雄的角色,也可以在另一部电影里扮演反派的角色。

DCI认为,对Role的建模应该是面向Context的,因为特定的业务行为只有在特定的业务场景下才会有意义。通过对Role的建模,我们就能够将领域对象的方法拆分出去,从而避免了上帝类的出现。最后,领域对象通过组合或继承的方式将Role集成起来,从而具备了扮演角色的能力。

图片

DCI架构一方面通过角色扮演模型使得领域模型易于理解,另一方面通过“ 小类大对象 ”的手法避免了上帝类的问题,从而较好地解决了贫血模型和充血模型之争。另外,将领域对象的行为根据Role拆分之后,模块更加的高内聚、低耦合了。

使用DCI建模

回到前面的案例,使用DCI的建模思路,我们可以将“人”的几种行为按照不同的角色进行划分。吃完、睡觉、玩游戏,是作为人类角色的行为;学习、考试,是作为学生角色的行为;上班、下班,是作为员工角色的行为;购票、游玩,则是作为游玩者角色的行为。“人”在这个场景中,充当的是人类的角色;在学校这个场景中,充当的是学生的角色;在公司这个场景中,充当的是员工的角色;在公园这个场景中,充当的是游玩者的角色。

图片

需要注意的是,学生、员工、游玩者,这些角色都应该具备人类角色的行为,比如在学校里,学生也需要吃饭。

最后,根据DCI建模出来的模型,应该是这样的:

图片

在DCI模型中,People不再是一个包含众多属性和方法的“上帝类”,这些属性和方法被拆分到多个Role中实现,而People由这些Role组合而成。

另外,SchoolCompany也不再耦合,School只引用了Student,不能调用与Company相关的WorkerWork()OffWorker()方法。

图片

代码实现DCI模型

DCI建模后的代码目录结构如下;

- context: 场景
  - company.go
  - home.go
  - park.go
  - school.go
- object: 对象
  - people.go
- data: 数据
  - account.go
  - identity_card.go
  - student_card.go
  - work_card.go
- role: 角色
  - enjoyer.go
  - human.go
  - student.go
  - worker.go

从代码目录结构上看,DDD和DCI架构相差并不大,aggregate目录演变成了context目录;vo目录演变成了data目录;entity目录则演变成了objectrole目录。

首先,我们实现基础角色HumanStudentWorkerEnjoyer都需要组合它:

package role

// 人类角色
type Human struct {
 data.IdentityCard
 data.Account
}
func (h *Human) Eat() {
 fmt.Printf("%+v eating\\n", h.IdentityCard)
 h.Account.Balance--
}
func (h *Human) Sleep() {
 fmt.Printf("%+v sleeping\\n", h.IdentityCard)
}
func (h *Human) PlayGame() {
 fmt.Printf("%+v playing game\\n", h.IdentityCard)
}

接着,我们再实现其他角色,需要注意的是, StudentWorkerEnjoyer不能直接组合Human ,否则People对象将会有4个Human子对象,与模型不符:

// 错误的实现
type Worker struct {
 Human
}
func (w *Worker) Work() {
 fmt.Printf("%+v working\\n", w.WorkCard)
 w.Balance++
}
...
type People struct {
 Human
 Student
 Worker
 Enjoyer
}
func main() {
 people := People{}
  fmt.Printf("People: %+v", people)
}
// 结果输出, People中有4个Human:
// People: {Human:{} Student:{Human:{}} Worker:{Human:{}} Enjoyer:{Human:{}}}
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 编程语言
    +关注

    关注

    10

    文章

    1945

    浏览量

    34760
  • 应用程序
    +关注

    关注

    37

    文章

    3273

    浏览量

    57727
  • DCI
    DCI
    +关注

    关注

    0

    文章

    39

    浏览量

    6835
  • 面向对象编程

    关注

    0

    文章

    22

    浏览量

    1815
收藏 人收藏

    评论

    相关推荐

    DCI 颠覆光器件产业?

    从不成熟到成熟的推进剂,但是DCI肯定要为此付出一点代价!关于技术架构DCI早先采用一种分布式架构。但是不知道为什么DCI现在热衷CWDM
    发表于 02-08 15:53

    Xilinx FPGA DCI使用方法

    各位大神,请问Xilinx FPGADCI是如何使用的?我知道是把每个Bank的VRP、VRN管脚分别下拉、上拉,除此之外,在HDL代码和约束应该如何写呢?查了半天资料没有查到,所以来论坛问问。@LQVSHQ
    发表于 08-20 20:51

    DDR3控制器和SSTL15_T_DCI在同一个bank

    你好,我使用Virtex7的HP库来实现DDR3控制器。我的控制器将以1600Mbps的速度运行,因此主控制器的VRN和VRP应连接一个80Ω电阻,以实现更高的性能。实现addr /
    发表于 03-25 11:04

    为什么银行也没有DCI匹配?

    在ml_605的示意图中,我发现在一个银行(例如银行16)混合了LVDS信号和信号端信号,所以银行应该收起2.5v,并且银行有DCI匹配。但是在银行24(银行混合了LVDS信号和信号端信号),所以
    发表于 10-25 08:47

    如何在IBIS文件配置SSTL135 DCI阻抗

    用作输入时并联施加GND和PWR设置?如何为我的电路板设计中使用的较低DCI阻抗正确配置IBIS文件?有人建议在设计添加外部终端电阻以进行仿真。但是,对于这些双向线路,DCI仅在充当输入时才有效。那些外部电阻会在输出期间产生负
    发表于 07-14 09:10

    DCI是什么?Xilinx 7系列FPGA的HP bank都支持DCI

    Xilinx 7系列FPGA的HP bank都支持DCI,目的是在高速单板信号传输中保持信号完整性,减少反射等因素影响,那么DCI是什么?digitally controlled impedance
    发表于 06-27 09:11 2w次阅读
    <b class='flag-5'>DCI</b>是什么?Xilinx 7系列FPGA的HP bank都支持<b class='flag-5'>DCI</b>

    一种AUTOSAR软件架构RTE的实现方法

    介绍了一种AUTOSAR软件架构RTE的实现方法。
    发表于 07-13 16:02 6次下载

    DCI BOX与传统WDM/OTN设备有什么区别?

    DCI-BOX,中国联通叫模块化波分设备,中国电信叫盒式波分设备DCI-BOX,是数据中心点到点互连(DCI)的设备。在DCI BOX出现之前,DC
    的头像 发表于 03-26 14:29 1786次阅读

    DCI BOX与传统WDM/OTN设备有什么区别?

    DCI BOX出现之前,DCI通常使用WDM/OTN设备进行互连,那么两者之间有什么区别呢?
    的头像 发表于 03-27 15:37 1155次阅读

    易飞扬全新升级DCI BOX,照亮DCI传输网络

    易飞扬2U 6.4T DCI BOX
    的头像 发表于 04-14 17:46 1150次阅读
    易飞扬全新升级<b class='flag-5'>DCI</b> BOX,照亮<b class='flag-5'>DCI</b>传输网络

    易飞扬非相干DCI BOX的DCI传输方案介绍

    易飞扬推出的最新1U 800G DWDM DCI BOX是一款1U盒式的多业务波分传输平台,可满足最大8×100GE业务点对点传输的应用场景,单机框常规接入容量800G。它同样满足DCI对小体积、低功耗、极简维护、低时延、大带宽的需求。
    发表于 04-21 10:39 554次阅读

    非相干DCI BOX,提供更经济的DCI传输方案

    易飞扬推出的最新1U 800G DWDM DCI BOX是一款1U盒式的多业务波分传输平台,可满足最大8×100GE业务点对点传输的应用场景,单机框常规接入容量800G。它同样满足DCI对小体积、低功耗、极简维护、低时延、大带宽的需求。
    的头像 发表于 04-21 10:40 888次阅读

    非相干DCI BOX,提供更经济的DCI传输方案

    上文,我们介绍了相干DCI BOX完美适配目前DCI传输的应用,不过,相干子系统的成本向来比较高,那是否有成本较低的非相干设备可供选择?考虑到不同用户的预算需求,易飞扬同样提供经济型的非相干DCI BOX,本文介绍的1U 800
    的头像 发表于 04-24 09:46 897次阅读
    非相干<b class='flag-5'>DCI</b> BOX,提供更经济的<b class='flag-5'>DCI</b>传输方案

    如何实现DCI架构(上)

    在面向对象编程的理念里,应用程序是对现实世界的抽象,我们经常会将现实的事物建模为编程语言中的类/对象(“ **是什么** ”),而事物的行为则建模为方法(“ **做什么** ”)。面向对象编程有
    的头像 发表于 05-10 17:09 706次阅读
    如何<b class='flag-5'>实现</b><b class='flag-5'>DCI</b><b class='flag-5'>架构</b>(上)

    如何实现DCI架构(下)

    在面向对象编程的理念里,应用程序是对现实世界的抽象,我们经常会将现实的事物建模为编程语言中的类/对象(“ **是什么** ”),而事物的行为则建模为方法(“ **做什么** ”)。面向对象编程有
    的头像 发表于 05-10 17:10 590次阅读