来源 | https://juejin.cn/post/6917125801460629518

  • 贫血模型即事务脚本模式

  • 充血模型即领域模型模式

贫血模型

最早广泛应用源于EJB2,最强盛时期则是由Spring创造,把

  • “行为”(逻辑、过程)

  • “状态”(数据,对应到语言就是对象成员变量)

分离到不同的对象中:

  • 只有状态的对象就是所谓的“贫血对象”(常称为VO——Value Object)

  • 只有行为的对象就是我们常见的N层结构中的Logic/Service/Manager层(对应到EJB2中的Stateless Session Bean)。(曾经Spring的作者Rod Johnson也承认,Spring不过是在沿袭EJB2时代的“事务脚本”,也就是面向过程编程)

贫血领域模型是一个存在已久的反模式,目前仍有许多拥趸者。Martin Fowler曾经和Eric Evans聊天谈到它时,都觉得这个模型似乎越来越流行了。作为领域模型的推广者,他们觉得这不是一件好事。

贫血领域模型的基本特征是:它第一眼看起来还真像这么回事儿。项目中有许多对象,它们的命名都是根据领域来的。对象之间有着丰富的连接方式,和真正的领域模型非常相似。但当你检视这些对象的行为时,会发现它们基本上没有任何行为,仅仅是一堆getter/setter。其实这些对象在设计之初就被定义为只能包含数据,不能加入领域逻辑。这些逻辑要全部写入一组叫Service的对象中。这些Service构建在领域模型之上,使用这些模型来传递数据。

这种反模式的恐怖之处在于,它完全和面向对象设计背道而驰。面向对象设计主张将数据和行为绑定在一起,而贫血领域模型则更像是一种面向过程设计,Martin Fowler和Eric在Smalltalk时就极力反对这种做法。更糟糕的时,很多人认为这些贫血领域对象是真正的对象,从而彻底误解了面向对象设计的涵义。

如今,面向对象的概念已经传播得很广泛了,而要反对这种贫血领域模型的做法,还需要更多论据。贫血领域模型的根本问题在于,它引入了领域模型设计的所有成本,却没有带来任何好处。最主要的成本是将对象映射到数据库中,从而产生了一个O/R(对象关系)映射层。只有当你充分使用了面向对象设计来组织复杂的业务逻辑后,这一成本才能够被抵消。如果将所有行为都写入到Service对象,那最终你会得到一组事务处理脚本,从而错过了领域模型带来的好处。正如martin在企业应用架构模式一书中说到的,领域模型并不一定是最好的工具。

将行为放入领域模型,这点和分层设计(领域层、持久化层、展现层等)并不冲突。因为领域模型中放入的是和领域相关的逻辑——验证、计算、业务规则等。如果你要讨论能否将数据源或展现逻辑放入到领域模型中,这就不在本文论述范围之内了。

一些面向对象专家的观点有时会让人产生疑惑,他们认为的确应该有一个面向过程的服务层。但是,这并不意味着领域模型就不应该包含行为。事实上,service层需要和一组富含行为的领域模型结合使用。

Eric Evans的Domain Driven Design一书中提到:

  • 应用层(即Service层)

描述应用程序所要做的工作,并调度丰富的领域模型来完成它。这个层次的任务是描述业务逻辑,或和其它项目的应用层做交互。这层很薄,不包含任何业务规则或知识,仅用于调度和派发任务给下一层的领域模型。这层没有业务状态,但可以为用户或程序提供任务状态

  • 领域层(或者叫模型层)

表示业务逻辑、业务场景和规则。该层次会控制和使用业务状态,即使这些状态最终会交由持久化层来存储。总之,该层是软件核心。

服务层很薄——所有重要的业务逻辑都写在领域层。他在服务模式中复述了这一观点:如今人们常犯的错误是不愿花时间将业务逻辑放到合适的领域模型中,从而逐渐形成面向过程的程序设计。

我不清楚为什么这种反模式会那么常见。我怀疑是因为大多数人并没有使用过一个设计良好的领域模型,特别是那些以数据为中心的开发人员。此外,有些技术也会推动这种反模式,比如J2EE的Entity Bean,这会让我更倾向于使用POJO领域模型。

总之,如果你将大部分行为都放置在服务层,那么你就会失去领域模型带来的好处。如果你将所有行为都放在服务层,那你就无可救药了。

优点

简单

  • 对于只有少量业务逻辑的应用来说,使用起来非常自然

  • 开发迅速,易于理解

  • 注意:也不能完全排斥这种方式

缺点

无法良好的应对复杂逻辑

  • 比如收入确认规则发生变化,例如在4月1号之前签订的合同要使用某规则.....

  • 和欧洲签订的合同使用另外一个规则.....

充血模型

面向对象设计的本质:“一个对象是拥有状态和行为的”,比如一个人:

  • 他眼睛什么样鼻子什么样这就是状态

  • 人可以去打游戏或是写程序,这就是行为

为什么要有一个“人Manager”这样的东西存在去帮人“打游戏”呢?举个简单的J2EE案例,设计一个与用户(User)相关功能。传统的设计一般是:

  • 类:User+UserManager

  • 保存用户调用:userManager.save(User user)

充血的设计则可能会是:

  • 类:User

  • 保存用户调用:user.save()

  • User有一个行为是:保存它自己

其实它们没有什么特别适用的方向,个人更倾向于总是使用充血模型,因为OOP总是比面向过程编程要有更丰富的语义、更合理的组织、更强的可维护性—当然也更难掌握。因此实际工程场景中,是否使用,如何使用还依赖于设计者以及团队充血模型设计的理解和把握,因为现在绝大多数J2EE开发者都受贫血模型影响非常深。另外,实际工程场景中使用充血模型还会碰到很多很多细节问题,其中最大的难关就是“如何设计充血模型”或者说“如何从复杂的业务中分离出恰到好处且包含语义的逻辑放到VO的行为中”。

如果一个对象包含其他对象,那就将职责继续委托下去,由具体的 POJO 执行业务逻辑,将策略模式更加细粒度,而不是写 ifelse。

往期推荐

Token多平台身份认证架构设计思路

955 不加班的公司名单!2021 年4月最新版!

Java API版权第一大案,索赔百亿美元,打了10年终于有结果了!

听说过OpenJDK,没说过OpenValueJDK吧?

2021年3月程序员工资统计数据出炉,又拖后腿了……

如果你喜欢本文,欢迎关注我,订阅更多精彩内容

关注我回复「加群」,加入Spring技术交流群

免费领取:大佬设计模式学习笔记

喜欢的这里报道

↘↘↘

什么是充血模型?什么又是贫血模型?相关推荐

  1. 贫血模型,充血模型(领域驱动设计)

    很多业务系统都是基于 MVC 三层架构来开发的.虽然这种开发模式已经成为标准的 Web 项目的开发模式,但它却违反了面向对象编程风格,是一种彻彻底底的面向过程的编程风格. MVC 三层架构中的 M 表 ...

  2. DDD(领域驱动设计)系列主题:失血模型,贫血模型,充血模型和胀血模型详细解读和代码案例说明!

    目录 失血模型,贫血模型,充血模型和胀血模型定义及优点和缺点 失血模型 贫血模型 充血模型 胀血模型 失血模型,贫血模型,充血和胀血代码样例 失血模型代码样例 贫血模型代码样例 充血模型代码样例 DD ...

  3. 一个简单例子:贫血模型or领域模型

    转:一个简单例子:贫血模型or领域模型 贫血模型 我们首先用贫血模型来实现.所谓贫血模型就是模型对象之间存在完整的关联(可能存在多余的关联),但是对象除了get和set方外外几乎就没有其它的方法,整个 ...

  4. DDD领域驱动设计 — 贫血模型与充血模型

    文章转载来源:https://juejin.cn/post/6917125801460629518 | 前言  要想深入掌握和了解 DDD 领域驱动设计的核心,那无论如何也绕不开两大较为抽象的概念-- ...

  5. DDD 领域驱动设计:贫血模型、充血模型的深入解读!

    作者:JavaEdge在掘金 链接:https://juejin.cn/post/6917125801460629518 -     前言     - 要想深入掌握和了解 DDD 领域驱动设计的核心, ...

  6. DDD 领域驱动设计:贫血模型、充血模型的深入解读

    点击上方"朱小厮的博客",选择"设为星标" 后台回复"书",获取 后台回复"k8s",可领取k8s资料 -     前言 ...

  7. 领域驱动设计-从贫血模型到充血模型

    背景 领域模型对象只是用来存储应用的数据.业务逻辑位于服务层中,管理域对象的数据.在服务层中,应用的每个实体对应一个服务类.这种模式大家是不是很熟悉,尤其是在中小项目或者项目刚启动的时候,都是怎么方便 ...

  8. 【设计模式之美】<Reading Notes>贫血模型与充血模型

    小知识 需要了解的一些名词 1.领域驱动设计(Domain Driven Design,简称 DDD) 2.MVC 三层架构 : M 表示 Model,V 表示 View,C 表示 Controlle ...

  9. 领域模型(domain model)贫血模型(anaemic domain model)充血模型(rich domain model)

    领域模型是领域内的概念类或现实世界中对象的可视化表示,又称为概念模型或分析对象模型,它专注于分析问题领域本身,发掘重要的业务领域概念,并建立业务领域概念之间的关系. 贫血模型是指使用的领域对象中只有s ...

最新文章

  1. 软件测试2019:第一次作业
  2. wxWidgets 的打印演示
  3. 在 Android 上使用协程(二):Getting started
  4. jQuery-1.9.1源码分析系列(二)jQuery选择器续2——筛选
  5. SAP Spartacus cxFocus增添了refresh Focus功能后的一些考虑
  6. 热像仪 二次开发 c++_重庆多功能红外线热像仪方案
  7. JavaScript导出图片和数据到Excel
  8. 每日编程-20170326
  9. 马斯克为何不惜激怒众“韭菜”?
  10. 20200210:(leetcode 623)在二叉树中增加一行
  11. qt中实现息屏开平mousepress_Qt元对象(Meta-Object)系统与反射
  12. KVM虚拟化实践-老男孩架构师课程教案笔记分享
  13. WPFの三种方式实现快捷键
  14. 使用X-shell管理员root连接ubuntu17.10服务器拒绝密码的一个失误!
  15. 计算机磁盘无法创建文件夹,无法创建文件,小编告诉你无法新建文件夹怎么办...
  16. 重新注册所有dll文件
  17. 使用git上传uni-app项目到Gitee
  18. 推荐Linux音乐播放器:咪咕音乐
  19. 安装Office InfoPath 2007
  20. 2015年最新国内十大应用商店广告报价表

热门文章

  1. 听研二师兄师姐报告收获
  2. linux c 通过 pid 获取 进程相关信息 cmdline
  3. python jenkins库 api简介
  4. tcp时间戳 引起的网站不能访问
  5. ndpi 流量协议分析
  6. linux 内核 struct file 获取文件名 全路径
  7. Ntop性能提升方案
  8. centos transmission 无法开启登录验证
  9. 使用__FILE__和__LINE__定位错误
  10. 学习LOWORD、 HIWORD、LOBYTE、HIBYTE