设计模式十二之组合模式

  在现实生活中,存在很多“部分-整体”的关系,例如,大学中的部门与学院、总公司中的部门与分公司、学习用品中的书与书包、生活用品中的衣月艮与衣柜以及厨房中的锅碗瓢盆等。在软件开发中也是这样,例如,文件系统中的文件与文件夹、窗体程序中的简单控件与容器控件等。对这些简单对象与复合对象的处理,如果用组合模式来实现会很方便。

1. 模式的定义与特点

1.1 模式的定义

  组合模式(Composite):有时又叫做“部分与整体模式“,是一种将对象整合成树状层次结构的模式,用来表示部分与整体的关系,使用户用单个对象和组合对象具有一致的访问性。

1.2 模式的特点

  组合模式的优点有:
    1. 使客户端代码可以一致地处理单个对象和组合对象,无需关系自己处理的是单个对象或是组合对象,简化了客户端代码;
    2. 更容易在组合体内加入新的对象,客户端不会因为加入新的对象而更改代码,满足开闭原则。

  组合模式的缺点有:
    1. 限制类型时较为复杂;
    2. 使设计变得更加抽象,客户端需要花更多的时间理清类之间的层次关系。

1.3 模式的使用场景

  1. 希望客户端可以忽略组合对象和单个对象的差异时;
  2. 处理一个树形结构时。

2. 模式的结构与实现

2.1 模式的结构

  组合模式的主要角色如下:
    1. 抽象构建角色(Component):主要为树叶构建角色和树枝构建角色声明公共接口,并实现他们的默认行为,在透明式的组合模式中抽象构建还声明和管理子类的接口;在安全式的组合模式中抽象构建不声明和管理子类的接口,管理工作由树枝构建完成;
    2. 树枝构建角色(Composite):是组合中的分支节点对象,它有子节点。它实现了抽象构建角色中声明的接口,主要作用是存储和管理子部件,通常包含 add() remove() getChild() 等方法。
    3. 树叶构建角色(Leaf):是组合中的叶子节点对象,它没有子节点,实现了抽象构建中声明的接口。

  组合模式分为透明式的组合模式和安全式的组合模式:
    1. 透明方式:在该方式中,由于抽象构件声明了所有子类中的全部方法,所以客户端无须区别树叶对象和树枝对象,对客户端来说是透明的。但其缺点是:树叶构件本来没有 Add()、Remove() 及 GetChild() 方法,却要实现它们(空实现或抛异常),这样会带来一些安全性问题;
透明式的组合模式类UML

    2. 安全方式:在该方式中,将管理子构件的方法移到树枝构件中,抽象构件和树叶构件没有对子对象的管理方法,这样就避免了上一种方式的安全性问题,但由于叶子和分支有不同的接口,客户端在调用时要知道树叶对象和树枝对象的存在,所以失去了透明性。

2.2 模式的实现

抽象构建

/**
 * 抽象构建 - 目录组件
 */
public abstract class CatalogComponent {

    public void add(CatalogComponent catalogComponent) {
        throw new UnsupportedOperationException("不支持添加操作");
    }

    public void remove(CatalogComponent catalogComponent) {
        throw new UnsupportedOperationException("不支持删除操作");
    }

    public String getName() {
        throw new UnsupportedOperationException("不支持获取名称操作");
    }

    public Double getPrice() {
        throw new UnsupportedOperationException("不支持获取价格操作");
    }

    public void print() {
        throw new UnsupportedOperationException("不支持打印操作");
    }

}

树枝构建

/**
 * 树枝构建 - 课程目录
 */
public class CourseCatalog extends CatalogComponent {

    private String name;

    private Integer level;

    private List<CatalogComponent> items = new ArrayList<>();

    public CourseCatalog(String name, Integer level) {
        this.name = name;
        this.level = level;
    }

    @Override
    public void add(CatalogComponent catalogComponent) {
        items.add(catalogComponent);
    }

    @Override
    public void remove(CatalogComponent catalogComponent) {
        items.remove(catalogComponent);
    }

    @Override
    public String getName() {
        return this.name;
    }

    @Override
    public void print() {
        System.out.println(this.name);
        for (CatalogComponent catalogComponent : items) {
            if (level != null) {
                for (int i=0; i<this.level; i++) {
                    System.out.print("  ");
                }
            }
            catalogComponent.print();
        }

    }
}

树叶构建

/**
 * 树叶构建 - 课程
 */
public class Course extends CatalogComponent {

    private String name;

    private Double price;

    public Course(String name, Double price) {
        this.name = name;
        this.price = price;
    }

    @Override
    public String getName() {
        return this.name;
    }

    @Override
    public Double getPrice() {
        return this.price;
    }

    @Override
    public void print() {
        System.out.println("课程名称:" + this.name + ", 课程价格为:" + this.price);
    }
}

客户端

public class Client {

    public static void main(String[] args) {
        CatalogComponent root = new CourseCatalog("学习目录", 1);
        CatalogComponent python = new Course("python 课程", 11D);
        CatalogComponent lunix = new Course("lunix 课程", 66D);

        CatalogComponent javaCatalog = new CourseCatalog("Java 学习", 2);
        CatalogComponent web = new Course("web 基础", 33D);
        CatalogComponent redis = new Course("redis 基础", 44D);
        CatalogComponent java = new Course("java 基础", 54D);

        root.add(python);
        root.add(lunix);
        javaCatalog.add(web);
        javaCatalog.add(redis);
        javaCatalog.add(java);
        root.add(javaCatalog);

        root.print();
    }
}

# 运行结果如下:
学习目录
  课程名称:python 课程, 课程价格为:11.0
  课程名称:lunix 课程, 课程价格为:66.0
  Java 学习
    课程名称:web 基础, 课程价格为:33.0
    课程名称:redis 基础, 课程价格为:44.0
    课程名称:java 基础, 课程价格为:54.0

3. 模式在开源软件中的应用

3.1 java.util.HashMap 类

public class HashMap<K,V> extends AbstractMap<K,V> implements Map<K,V>, Cloneable, Serializable {
    public void putAll(Map<? extends K, ? extends V> m) {
        putMapEntries(m, true);
    }
}

3.2 java.util.ArrayList 类

public class ArrayList<E> extends AbstractList<E> implements List<E>, RandomAccess, Cloneable, java.io.Serializable {
            
    public boolean addAll(Collection<? extends E> c) {
        Object[] a = c.toArray();
        int numNew = a.length;
        ensureCapacityInternal(size + numNew);  // Increments modCount
        System.arraycopy(a, 0, elementData, size, numNew);
        size += numNew;
        return numNew != 0;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值