【设计模式】【结构型模式】组合模式(Composite)

ops/2025/2/22 0:14:43/

👋hi,我不是一名外包公司的员工,也不会偷吃茶水间的零食,我的梦想是能写高端CRUD
🔥 2025本人正在沉淀中… 博客更新速度++
👍 欢迎点赞、收藏、关注,跟上我的更新节奏
🎵 当你的天空突然下了大雨,那是我在为你炸乌云

文章目录

一、入门

什么是组合模式

组合模式(Composite Pattern)是一种结构型设计模式,它允许你将对象组合成树形结构来表示“部分-整体”的层次关系。组合模式使得客户端可以统一处理单个对象和组合对象,而无需关心它们的具体类型。

组合模式的核心思想是:将对象组织成树形结构,使得对单个对象和组合对象的操作具有一致性。

为什么有要组合模式

假设我们正在开发一个电商平台,需要管理商品的分类和展示。商品分类是一个典型的树形结构,比如:

  • 电子产品
    • 手机
      • 苹果手机
      • 安卓手机
    • 电脑
      • 笔记本电脑
      • 台式机
  • 服装
    • 男装
    • 女装

在这个系统中,每个分类(如“电子产品”)可以包含子分类(如“手机”),也可以包含具体的商品(如“苹果手机”)
不使用组合模式的实现

// 分类类
public class Category {private String name;private List<Category> subCategories = new ArrayList<>();private List<Product> products = new ArrayList<>();public Category(String name) {this.name = name;}public void addSubCategory(Category category) {subCategories.add(category);}public void addProduct(Product product) {products.add(product);}public void display() {System.out.println("分类: " + name);for (Category category : subCategories) {category.display();}for (Product product : products) {product.display();}}
}// 商品类
public class Product {private String name;public Product(String name) {this.name = name;}public void display() {System.out.println("商品: " + name);}
}

客户端代码

public class Client {public static void main(String[] args) {// 创建商品Product iphone = new Product("iPhone 15");Product macbook = new Product("MacBook Pro");// 创建子分类Category phones = new Category("手机");phones.addProduct(iphone);Category computers = new Category("电脑");computers.addProduct(macbook);// 创建顶级分类Category electronics = new Category("电子产品");electronics.addSubCategory(phones);electronics.addSubCategory(computers);// 展示分类结构electronics.display();}
}

存在问题

  • 代码重复CategoryProduct需要分别定义display方法。
  • 类型判断:如果需要统一处理分类和商品,客户端需要频繁判断对象类型。
  • 扩展性差:新增节点类型(如“品牌”)时,需要修改大量代码。
  • 难以维护:嵌套层次较深时,代码会变得混乱。

怎么样实现组合模式

组合模式的组成:
抽象组件(Component):

  • 作用:定义所有节点(叶子节点和复合节点)的通用接口或抽象类。
  • 职责
    • 声明所有节点共有的方法(如displayaddremove等)。
    • 提供默认实现(可选)。
  • 特点
    • 是叶子节点和复合节点的父类或接口。
    • 客户端通过抽象组件与具体节点交互。

叶子节点(Leaf)

  • 作用:表示树形结构中的叶子节点,没有子节点。
  • 职责
    • 实现抽象组件定义的方法。
    • 通常是树形结构中的最小单元。
  • 特点
    • 不能包含其他节点。
    • 组合模式中的“部分”。

复合节点(Composite)

  • 作用:表示树形结构中的复合节点,可以包含子节点(叶子节点或其他复合节点)。
  • 职责
    • 实现抽象组件定义的方法。
    • 管理子节点(如添加、删除、遍历子节点)。
  • 特点
    • 可以包含其他节点。
    • 组合模式中的“整体”。

【案例】电商分类 - 改
在这里插入图片描述
抽象组件(Component):定义所有节点(分类和商品)的通用行为。

public abstract class ProductCategory {protected String name;public ProductCategory(String name) {this.name = name;}public abstract void display();
}

叶子节点(Leaf):表示具体的商品,没有子节点。

public class Product extends ProductCategory {public Product(String name) {super(name);}@Overridepublic void display() {System.out.println("商品: " + name);}
}

复合节点(Composite):表示分类,可以包含子分类或商品。

import java.util.ArrayList;
import java.util.List;public class Category extends ProductCategory {private List<ProductCategory> children = new ArrayList<>();public Category(String name) {super(name);}public void add(ProductCategory category) {children.add(category);}@Overridepublic void display() {System.out.println("分类: " + name);for (ProductCategory child : children) {child.display();}}
}

客户端使用

public class Client {public static void main(String[] args) {// 创建商品ProductCategory iphone = new Product("iPhone 15");ProductCategory macbook = new Product("MacBook Pro");// 创建子分类Category phones = new Category("手机");phones.add(iphone);Category computers = new Category("电脑");computers.add(macbook);// 创建顶级分类Category electronics = new Category("电子产品");electronics.add(phones);electronics.add(computers);// 展示分类结构electronics.display();}
}// 结果
// 分类: 电子产品
// 分类: 手机
// 商品: iPhone 15
// 分类: 电脑
// 商品: MacBook Pro

二、总结

组合模式的优点

  1. 统一处理单个对象和组合对象
    • 核心优势:客户端无需区分处理的是叶子节点(单个对象)还是复合节点(组合对象)。
    • 案例:在文件系统中,无论是文件(叶子节点)还是文件夹(复合节点),都可以统一调用display()方法展示信息。
  2. 简化客户端代码
    • 核心优势:客户端不需要写复杂的条件判断(如if-else)来处理不同类型的节点。
    • 案例:在电商平台的商品分类系统中,客户端只需调用display()方法,无论是单品还是套餐都能自动递归处理。
  3. 扩展性强
    • 核心优势:新增节点类型时,只需继承抽象组件,符合开闭原则。
    • 案例:若要在组织架构中新增“项目组”类型,只需继承OrganizationUnit,不影响现有代码。
  4. 树形结构清晰
    • 核心优势:天然支持递归和嵌套,能直观表示“部分-整体”的层次关系。
    • 案例:UI框架(如Java Swing)通过组合模式构建组件树,方便渲染和事件传递。

组合模式的缺点

  1. 设计复杂性
    • 问题:需要抽象组件同时定义叶子节点和复合节点的行为,可能导致接口臃肿。
    • 案例:如果叶子节点(如File)需要实现复合节点的add()方法,只能抛出异常(UnsupportedOperationException),不够优雅。
  2. 类型系统的限制
    • 问题:客户端可能仍需通过instanceof检查类型,破坏透明性。
    • 案例:在文件系统中,若需要对目录(复合节点)执行特殊操作(如计算总大小),客户端可能被迫检查类型。
  3. 过度泛化
    • 问题:如果业务场景没有明确的层次结构,强行使用组合模式会导致设计过度复杂。
    • 案例:简单的商品列表不需要组合模式,直接使用集合即可。

组合模式的适用场景

  1. 树形结构的数据
    • 场景:需要表示对象的“部分-整体”层次关系。
    • 典型案例:
      • 文件系统(文件 + 文件夹)
      • UI组件树(按钮 + 面板)
      • 组织架构(员工 + 部门)
  2. 需要统一操作所有节点
    • 场景:希望对所有节点执行相同操作(如渲染、计算总价、权限校验)。
    • 典型案例:
      • 电商平台计算单品和套餐的总价格。
      • 游戏场景中渲染所有游戏对象(角色、道具、地图)。
  3. 动态组合嵌套结构
    • 场景:需要动态地添加、删除或嵌套节点。
    • 典型案例:
      • 菜单系统(支持菜单项和子菜单)。
      • 权限管理系统(角色可以包含多个权限,权限可以嵌套)。

http://www.ppmy.cn/ops/160365.html

相关文章

JavaScript 前端面试 1 (数据类型、数据结构)

一&#xff1a;JavaScript中有哪些数据类型&#xff1f;在存储上的差异&#xff1f; JavaScript的数据类型分为两类&#xff1a;基本类型 和 引用类型 两者的区别是存储位置不同 1&#xff1a;基本类型 基本类型有以下六种&#xff1a; Number数字类型&#xff0c;用于表示…

[数据结构]顺序表详解

目录 一.线性表 二.顺序表 2.1概念及结构 1. 静态顺序表&#xff1a;使用定长数组存储元素。 2. 动态顺序表&#xff1a;使用动态开辟的数组存储。 2.1按需申请 2.2 接口实现&#xff1a;增删查改 SeqList.h: SeqList.c: test.c 一.线性表 线性表 &#xff08; line…

DeepSeek-R1使用生存指南

文章目录 1.为什么普通人一定要使用DeepSeek2.DeepSeek的几种使用方式2.1网页端直接使用2.2手机端app使用2.3其他第三方平台 3.网页端按钮的说明4.正确的提问技巧4.1不要定义过程4.2明确受众4.3记忆时间有限4.4输出长度限制4.5如何清除上下文的记忆 5.几个避坑点5.1冗长提示词污…

国产编辑器EverEdit - 独门暗器:自动监视剪贴板内容

1 监视剪贴板 1.1 应用场景 如果需要对剪贴板的所有历史进行记录&#xff0c;并进行分析和回顾&#xff0c;则可以使用监视剪贴板功能&#xff0c;不仅在EverEdit中的复制会记录&#xff0c;在其他应用的复制也会记录。 1.2 使用方法 新建一个空文档(重要&#xff1a;防止扰乱…

Ubuntu22 server 安装 Chrome浏览器

前言 在 Ubuntu 22 上安装 Google Chrome 浏览器通常会遇到一些问题&#xff0c;尤其是在没有图形界面的服务器版本上。本文将详细介绍如何通过命令行使用 apt 安装 Google Chrome&#xff0c;并解决 GPG 公钥错误等常见问题。 目录 前言1. 安装前的准备2. 导入 Google 的 GPG…

利用Java爬虫精准获取商品销量详情:实战案例指南

在电商领域&#xff0c;商品销量数据是衡量产品受欢迎程度和市场表现的关键指标。精准获取商品销量详情不仅能帮助商家优化产品策略&#xff0c;还能为市场研究和数据分析提供丰富的数据资源。本文将详细介绍如何利用Java爬虫技术精准获取商品销量详情&#xff0c;并分享关键技…

Unity面板介绍_层级面板(23.1.1)

一、Inspector(检视面板) 显示当前选定游戏对象附加的组件及其属性信息。为重要游戏物体选择图标 二、面板详情

DeepSeek 新注意力架构NSA

DeepSeek 新注意力架构NSA概要 研究背景&#xff1a; 实现高效长上下文建模的自然方法是利用 softmax 注意力的固有稀疏性&#xff0c;通过选择性计算关键 query-key 对&#xff0c;可以显著减少计算开销&#xff0c;同时保持性能。最近这一路线的进展包括多种策略&#xff1…