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

简介: 一、入门 什么是组合模式 组合模式(Composite Pattern)是一种结构型设计模式,它允许你将对象组合成树形结构来表示“部分-整体”的层次关系。组合模式使得客户端可以统一处理单个对象和组合对

👋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)

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

【案例】电商分类 - 改
image.png

抽象组件(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);
    }

    @Override
    public 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);
    }

    @Override
    public 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. 动态组合嵌套结构
    • 场景:需要动态地添加、删除或嵌套节点。
    • 典型案例:
      • 菜单系统(支持菜单项和子菜单)。
      • 权限管理系统(角色可以包含多个权限,权限可以嵌套)。
目录
相关文章
|
6月前
|
设计模式 存储 缓存
【设计模式】【结构型模式】享元模式(Flyweight)
一、入门 什么是享元模式? 享元模式(Flyweight Pattern)是一种结构型设计模式,旨在通过共享对象来减少内存使用,特别适用于存在大量相似对象的情况。 它的核心思想是将对象的内在状态(不变
251 16
|
6月前
|
设计模式 Java 数据库连接
【设计模式】【结构型模式】代理模式(Proxy)
一、入门 什么是代理模式? 代理模式(Proxy Pattern)是一种结构型设计模式,允许你提供一个代理对象来控制对另一个对象的访问。 代理对象在客户端和目标对象之间起到中介作用,可以在不改变目标对
176 10
|
6月前
|
关系型数据库 Java MySQL
【设计模式】【结构型模式】桥接模式(Bridge)
一、入门 什么是桥接模式? 桥接模式(Bridge Pattern)是一种结构型设计模式,核心思想是将抽象与实现分离,让它们可以独立变化。简单来说,它像一座“桥”连接了两个维度的变化,避免用继承导致代
386 10
|
6月前
|
设计模式 前端开发 Java
【设计模式】【结构型模式】适配器模式(Adpter)
一、入门 什么是适配器模式? 适配器模式是Java中常用的结构型设计模式,它的核心作用就像现实中的电源转换器一样---让原本不兼容的两个接口能够协同工作。 为什么要用适配器模式? 假设我们需要在电商系
172 10
|
6月前
|
设计模式 Java 数据库连接
【设计模式】【结构型模式】外观模式(Facde)
一、入门 什么是外观模式? 一种结构型设计模式,通过为子系统中的一组接口提供一个统一的高层接口(称为外观),来简化客户端与复杂子系统的交互过程。其本质是建立抽象层来隔离复杂度。 为什么要有外观模式?
287 9
|
6月前
|
设计模式 缓存 安全
【设计模式】【结构型模式】装饰者模式(Decorator)
一、入门 什么是装饰者模式? 装饰者模式(Decorator Pattern)是 Java 中常用的结构型设计模式,它能在不修改原有对象结构的前提下,动态地为对象添加额外的职责。 为什么要装饰者模式?
161 8
|
9月前
|
设计模式 存储 安全
「全网最细 + 实战源码案例」设计模式——组合模式
组合模式(Composite Pattern)是一种结构型设计模式,用于将对象组合成树形结构以表示“部分-整体”的层次结构。它允许客户端以一致的方式对待单个对象和对象集合,简化了复杂结构的处理。组合模式包含三个主要组件:抽象组件(Component)、叶子节点(Leaf)和组合节点(Composite)。通过这种模式,客户端可以统一处理简单元素和复杂元素,而无需关心其内部结构。适用于需要实现树状对象结构或希望以相同方式处理简单和复杂元素的场景。优点包括支持树形结构、透明性和遵循开闭原则;缺点是可能引入不必要的复杂性和过度抽象。
288 22
|
设计模式 Java
Java设计模式:组合模式的介绍及代码演示
组合模式是一种结构型设计模式,用于将多个对象组织成树形结构,并统一处理所有对象。例如,统计公司总人数时,可先统计各部门人数再求和。该模式包括一个通用接口、表示节点的类及其实现类。通过树形结构和节点的通用方法,组合模式使程序更易扩展和维护。
204 2
Java设计模式:组合模式的介绍及代码演示
|
设计模式 存储 安全
Java设计模式-组合模式(13)
Java设计模式-组合模式(13)
154 2
|
6月前
|
设计模式 Java 数据库连接
【设计模式】【创建型模式】工厂方法模式(Factory Methods)
一、入门 什么是工厂方法模式? 工厂方法模式(Factory Method Pattern)是一种创建型设计模式,它定义了一个用于创建对象的接口,但由子类决定实例化哪个类。工厂方法模式使类的实例化延迟
200 16