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

    文章

    1952

    浏览量

    35830
  • 应用程序
    +关注

    关注

    38

    文章

    3316

    浏览量

    58571
  • DCI
    DCI
    +关注

    关注

    0

    文章

    40

    浏览量

    6941
  • 面向对象编程

    关注

    0

    文章

    22

    浏览量

    1889
收藏 人收藏
    相关推荐
    热点推荐

    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 7次下载

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

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

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

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

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

    易飞扬2U 6.4T DCI BOX
    的头像 发表于 04-14 17:46 1337次阅读
    易飞扬全新升级<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 710次阅读

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

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

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

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

    如何实现DCI架构(上)

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

    如何实现DCI架构(下)

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

    半导体芯片需要做哪些测试

    首先我们需要了解芯片制造环节做⼀款芯片最基本的环节是设计->流片->封装->测试,芯片成本构成⼀般为人力成本20%,流片40%,封装35%,测试5%(对于先进工艺,流片成本可能超过60%)。测试其实是芯片各个环节中最“便宜”的一步,在这个每家公司都喊着“CostDown”的激烈市场中,人力成本逐年攀升,晶圆厂和封装厂都在乙方市场中“叱咤风云”,唯独只有测试显

    汉通达
    5小时前
    138

    解决方案 | 芯佰微赋能示波器:高速ADC、USB控制器和RS232芯片——高性能示波器的秘密武器!

    示波器解决方案总述:示波器是电子技术领域中不可或缺的精密测量仪器,通过直观的波形显示,将电信号随时间的变化转化为可视化图形,使复杂的电子现象变得清晰易懂。无论是在科研探索、工业检测还是通信领域,示波器都发挥着不可替代的作用,帮助工程师和技术人员深入剖析电信号的细节,精准定位问题所在,为创新与发展提供坚实的技术支撑。一、技术瓶颈亟待突破性能指标受限:受模拟前端

    芯佰微电子
    4小时前
    166

    硬件设计基础----运算放大器

    1什么是运算放大器运算放大器(运放)用于调节和放大模拟信号,运放是一个内含多级放大电路的集成器件,如图所示:左图为同相位,Vn端接地或稳定的电平,Vp端电平上升,则输出端Vo电平上升,Vp端电平下降,则输出端Vo电平下降;右图为反相位,Vp端接地或稳定的电平,Vn端电平上升,则输出端Vo电平下降,Vn端电平下降,则输出端Vo电平上升2运算放大器的性质理想运算

    张飞实战电子官方
    19小时前
    225

    ElfBoard技术贴|如何调整eMMC存储分区

    ELF 2开发板基于瑞芯微RK3588高性能处理器设计,拥有四核ARM Cortex-A76与四核ARM Cortex-A55的CPU架构,主频高达2.4GHz,内置6TOPS算力的NPU,这一设计让它能够轻松驾驭多种深度学习框架,高效处理各类复杂的AI任务。

    ElfBoard
    1天前
    478

    米尔基于MYD-YG2LX系统启动时间优化应用笔记

    1.概述MYD-YG2LX采用瑞萨RZ/G2L作为核心处理器,该处理器搭载双核Cortex-A55@1.2GHz+Cortex-M33@200MHz处理器,其内部集成高性能3D加速引擎Mail-G31GPU(500MHz)和视频处理单元(支持H.264硬件编解码),16位的DDR4-1600/DDR3L-1333内存控制器、千兆以太网控制器、USB、CAN、

    米尔电子
    1天前
    265

    运放技术——基本电路分析

    虚短和虚断的概念由于运放的电压放大倍数很大,一般通用型运算放大器的开环电压放大倍数都在80dB以上。而运放的输出电压是有限的,一般在10V~14V。因此运放的差模输入电压不足1mV,两输入端近似等电位,相当于“短路”。开环电压放大倍数越大,两输入端的电位越接近相等。“虚短”是指在分析运算放大器处于线性状态时,可把两输入端视为等电位,这一特性称为虚假短路,简称

    张飞实战电子官方
    1天前
    348

    飞凌嵌入式携手中移物联,谱写全国产化方案新生态

    4月22日,飞凌嵌入式“2025嵌入式及边缘AI技术论坛”在深圳成功举办。中移物联网有限公司(以下简称“中移物联”)携OneOS操作系统与飞凌嵌入式共同推出的工业级核心板亮相会议展区,操作系统产品部高级专家严镭受邀作《OneOS工业操作系统——助力国产化智能制造》主题演讲。

    飞凌嵌入式
    2天前
    812

    ATA-2022B高压放大器在螺栓松动检测中的应用

    实验名称:ATA-2022B高压放大器在螺栓松动检测中的应用实验方向:超声检测实验设备:ATA-2022B高压放大器、函数信号发生器,压电陶瓷片,数据采集卡,示波器,PC等实验内容:本研究基于振动声调制的螺栓松动检测方法,其中低频泵浦波采用单频信号,而高频探测波采用扫频信号,利用泵浦波和探测波在接触面的振动声调制响应对螺栓的松动程度进行检测。通过螺栓松动检测

    Aigtek安泰电子
    2天前
    1k

    MOS管驱动电路——电机干扰与防护处理

    此电路分主电路(完成功能)和保护功能电路。MOS管驱动相关知识:1、跟双极性晶体管相比,一般认为使MOS管导通不需要电流,只要GS电压(Vbe类似)高于一定的值,就可以了。MOS管和晶体管向比较c,b,e—–>d(漏),g(栅),s(源)。2、NMOS的特性,Vgs大于一定的值就会导通,适合用于源极接地时的情况(低端驱动),只要栅极电压达到4V或10V就可以

    张飞实战电子官方
    2天前
    390

    压敏(MOV)在电机上的应用剖析

    一前言有刷直流电机是一种较为常见的直流电机。它的主要特点包括:1.结构相对简单,由定子、转子、电刷和换向器等组成;2.通过电刷与换向器的接触来实现电流的换向,从而使电枢绕组中的电流方向周期性改变,保证电机持续运转;3.具有调速性能较好等优点,可以通过改变电压等方式较为方便地调节转速。有刷直流电机在许多领域都有应用,比如一些电动工具、玩具、小型机械等。但它也存

    深圳市韬略科技有限公司
    05-06 11:34
    265

    硬件原理图学习笔记

    这一个星期认真学习了硬件原理图的知识,做了一些笔记,方便以后查找。硬件原理图分为三类1.管脚类(gpio)和门电路类输入输出引脚,上拉电阻,三极管与门,或门,非门上拉电阻:正向标志作用,给悬空的引脚一个确定的状态三极管:反向三极管(gpio输出高电平,NP两端导通,被控制端导通,电压为0)->NPN正向三极管(gpio输出低电平,PN两端导通,被控制端导通,

    张飞实战电子官方
    04-30 18:40
    454

    TurMass™ vs LoRa:无线通讯模块的革命性突破

    TurMass™凭借其高传输速率、强大并发能力、双向传输、超强抗干扰能力、超远传输距离、全国产技术、灵活组网方案以及便捷开发等八大优势,在无线通讯领域展现出强大的竞争力。

    道生物联
    05-06 10:50
    839

    RZT2H CR52双核BOOT流程和例程代码分析

    RZT2H是多核处理器,启动时,需要一个“主核”先启动,然后主核根据规则,加载和启动其他内核。本文以T2H内部的CR52双核为例,说明T2H多核启动流程。

    RA生态工作室
    04-03 17:14
    2.1k

    干簧继电器在RF信号衰减中的应用与优势

    在电子测试领域,RF(射频)评估是不可或缺的一部分。无论是研发阶段的性能测试,还是生产环节的质量检测,RF测试设备都扮演着关键角色。然而,要实现精准的RF评估,测试设备需要一种特殊的电路——衰减电路。这些电路的作用是调整RF信号的强度,以便测试设备能够准确地评估RF组件和RF电路的各个方面。衰减器的挑战衰减器的核心功能是校准RF信号的强度。为了实现这一点,衰

    斯丹麦德电子
    04-30 11:33
    743

    ElfBoard嵌入式教育科普|ADC接口全面解析

    当代信息技术体系中,嵌入式系统接口作为数据交互的核心基础设施,构成了设备互联的神经中枢。基于标准化通信协议与接口规范的技术架构,实现了异构设备间的高效数据交换与智能化协同作业。本文选取模数转换接口ADC作为技术解析切入点,通过系统阐释其工作机理、性能特征及重要参数,为嵌入式学习者爱好者构建全维度接口技术认知框架。

    ElfBoard
    04-30 09:34
    405