当前位置: 首页 > news >正文

贫血模型与充血模型

贫血模型与充血模型

  • 显著差异。
    • 贫血模型将业务逻辑分散在服务层(Service)。
    • 充血模型则将业务逻辑内聚在实体类(Entity)中。

特性对比

特性贫血模型 (Anemic Domain Model)充血模型 (Rich Domain Model)
业务逻辑位置主要在服务层 (Service Layer)内聚在实体类 (Entity Class)
数据结构仅包含数据属性 (Getters/Setters)包含业务方法 (Methods)
代码可读性较低,业务逻辑分散较高,业务逻辑集中
维护性较差,逻辑分散在多个类中较好,逻辑集中在实体类中
扩展性较低,需要添加新的实体类较高,添加新的实体类不会影响其他类
领域事件不支持或难以实现支持,易于实现领域事件
领域模型设计不符合领域驱动设计 (DDD)符合领域驱动设计 (DDD)
测试性较差,业务逻辑分散较好,业务逻辑集中
代码重复率较高,服务层可能重复逻辑较低,业务逻辑集中在实体类中
学习曲线较低,简单易懂较高,需要理解领域驱动设计概念

案例 - 用户转账

贫血模型写法
// 贫血的订单实体 (Anemic Order Entity)
public class Order {private String orderId;private double amount;private String status; // 例如: "PENDING", "PAID", "SHIPPED", "CANCELLED"// 只有 getter 和 setter 方法public String getOrderId() { return orderId; }public void setOrderId(String orderId) { this.orderId = orderId; }public double getAmount() { return amount; }public void setAmount(double amount) { this.amount = amount; }public String getStatus() { return status; }public void setStatus(String status) { this.status = status; }// 没有业务方法,比如 order.pay() 或 order.cancel()
}// 订单服务 (OrderService) - 业务逻辑都在这里
public class OrderService {public void createOrder(Order order) { /* 保存订单到数据库 */ }public void processPayment(String orderId) {Order order = orderRepository.findById(orderId); // 从数据库获取贫血 Order 对象if ("PENDING".equals(order.getStatus())) {// 执行支付逻辑order.setStatus("PAID"); // 修改状态orderRepository.save(order); // 保存回数据库} else {throw new IllegalStateException("订单状态不正确,无法支付");}}public void cancelOrder(String orderId) {Order order = orderRepository.findById(orderId);if ("PENDING".equals(order.getStatus())) {// 执行取消逻辑order.setStatus("CANCELLED"); // 修改状态orderRepository.save(order);} else {throw new IllegalStateException("订单状态不正确,无法取消");}}
}
充血模型写法
// 充血的订单实体 (Rich Order Entity)
public class Order {private String orderId;private double amount;private OrderStatus status; // 使用枚举或更复杂的对象表示状态public Order(String orderId, double amount) {this.orderId = orderId;this.amount = amount;this.status = OrderStatus.PENDING; // 初始状态在构造时设定}// 业务方法内聚在实体中public void pay() {if (this.status != OrderStatus.PENDING) {throw new IllegalStateException("订单状态不正确,无法支付。当前状态: " + this.status);}// 执行支付相关的内部逻辑(例如扣款,如果实体内部能处理)this.status = OrderStatus.PAID; // 状态变更逻辑在实体内部// 也可以触发领域事件}public void cancel() {if (this.status != OrderStatus.PENDING && this.status != OrderStatus.PAID) {throw new IllegalStateException("订单状态不正确,无法取消。当前状态: " + this.status);}// 执行取消相关的内部逻辑this.status = OrderStatus.CANCELLED; // 状态变更逻辑在实体内部}// getter 方法public String getOrderId() { return orderId; }public double getAmount() { return amount; }public OrderStatus getStatus() { return status; }
}// 订单服务 (OrderService) - 变得更薄,只负责事务和协调
public class OrderService {public void processOrderPayment(String orderId) {Order order = orderRepository.findById(orderId); // 从数据库获取充血 Order 对象order.pay(); // 调用实体自身的业务方法orderRepository.save(order); // 保存更新后的实体}public void cancelCustomerOrder(String orderId) {Order order = orderRepository.findById(orderId);order.cancel(); // 调用实体自身的业务方法orderRepository.save(order);}
}public enum OrderStatus {PENDING, PAID, SHIPPED, CANCELLED
}

总结

充血模型更符合领域驱动设计的原则,能够提高代码的可读性、维护性和扩展性。

通过将业务逻辑集中在实体类中,充血模型能够更好地反映领域概念,并支持领域事件的实现。

充血模型虽然学习曲线较陡,但在复杂业务场景中能够提供更好的解决方案。

实际上很多项目都使用的贫血模型,无他,简单、快速。天然适合快速开发。

最后 充血模型其实更像是理想,贫血模型是现实

理想很丰满,现实很骨感

如果团队成员对领域驱动设计不熟悉,或者项目需求简单,还是使用贫血模型可以更快地上手和迭代。

如果项目复杂,业务逻辑较多,充血模型能够更好地组织代码和业务逻辑。

充血模型对团队开发者的要求较高,需要理解领域驱动设计的概念和实践,自己个人项目尝试差不多了。

相关文章:

  • 实现网页中嵌入B站视频播放器:解决high_quality=1 失效的问题
  • 03 - ECA模块
  • Python-Flask实现登录
  • XAttention 计算步骤详解及示例
  • 58、嵌入式Servlet容器-【源码分析】切换web服务器与定制化
  • UDS协议中0x34、0x36、0x37服务详解及应用
  • 表达式的自动类型转换
  • HashMap vs LinkedHashMap
  • 短剧热浪,席卷海内外。
  • 2025企业级BI产品评测和推荐
  • 论坛系统自动化测试
  • GRUB2 启动配置的工作原理与优先级规则详解
  • SSH远程连接到Windows服务器
  • 概率基础——不确定性的数学
  • 1.3、SDH光接口类型
  • 批处理实现:自动抓取perfetto日志 自动导出到当前文件夹 自动打开分析页面
  • (nice!!!)(LeetCode 每日一题) 2616. 最小化数对的最大差值 (二分查找)
  • 落地 DDD 领域模型(常见的实现模式)
  • 单项链表的操作及其实现
  • 【二叉树】(四)二叉搜索树的基础修改构造及属性求解1
  • 可以做全景的网站/营销宣传方案
  • 域名网站备案/郑州seo公司
  • vr全景网站开发制作/广告外链购买交易平台
  • 便宜做网站的公司靠谱吗/东莞今天最新消息新闻
  • 有哪些网站可以卖自己做的图片/惠州seo报价
  • 做网站的公司利润/郑州网站seo技术