组合模式的Java实现

组合模式简述

  • 组合模式(Composite Pattern),又叫部分整体模式,是用于把一组相似的对象当作一个单一的对象,从而使得某些操作具有一致性,在客户端角度来看,不需要针对部分与整体的特殊性进行分类处理,这种操作的一致性是通过组合模式在内部进行实现的。
  • 这里主要解决的问题是类似于树形结构的问题,在树形结构中,叶子结点就是部分,整体就是非叶子结点,整体中包含部分,可以理解为非叶子节点还会继续往下链接,直到链接到叶子结点。
  • 组合模式依据树形结构来组合对象,用来表示部分以及整体层次。
  • 这种类型的设计模式属于结构型模式,它创建了对象组的树形结构。
  • 这种模式创建了一个包含自己对象组的类,该类提供了修改相同对象组的方式。

介绍

意图:将对象组合成树形结构以表示"部分-整体"的层次结构,组合模式使得用户对单个对象和组合对象的使用具有一致性。

主要解决:它在我们树型结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以像处理简单元素一样来处理复杂元素,从而使得客户程序与复杂元素的内部结构解耦。

何时使用: 

  • 想表示对象的部分-整体层次结构(树形结构)。
  • 希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象。

组合模式核心:

  • 抽象构建角色(component):定义了叶子和容器构建的共同点
  • 叶子构建角色(leaf):无子节点
  • 容器构建角色(composite):有容器特征,可以包含子节点
  • 树枝(容器)和叶子实现统一接口,树枝内部组合该接口

关键代码:树枝内部组合该接口,并且含有内部属性 List,里面放 Leaf,展示如下

package com.ly.composite;

/**
 * liyang 2020-07-20
 * 组合模式的组织形式,仅用于展示
 * 实际使用是叶子组件与容器组件去实现Component
 * 具体见下文的案例
 */

public interface Component {
    void operation();
}

//叶子组件
interface Leaf extends Component {

}

//容器组件
interface Composite extends Component {
    //List(Leaf)
    void add(Component c);
    void remove(Component c);
    Component getChild(int index);
}

组合模式工作流程分析:

  • 组合模式为处理树形结构提供了完美的解决方案,描述了如何将容器和叶子进行递归组合,使得用户在使用时可以一致的对待容器和叶子。
  • 当容器方法被调用时,经遍历整个树形结构,寻找也包含这个方法的成员,并调用执行。其中,使用了递归调用的机制对整个结构进行处理。

优点:

  • 高层模块调用简单
  • 节点自由增加。

缺点:在使用组合模式时,其叶子和树枝的声明都是实现类,而不是接口,违反了依赖倒置原则。

使用场景:部分、整体场景,如树形菜单,文件、文件夹的管理。

应用实例:

  • 算术表达式包括操作数、操作符和另一个操作数,其中,另一个操作符也可以是操作数、操作符和另一个操作数。
  • 在 JAVA AWT 和 SWING 中,对于 Button 和 Checkbox 是树叶,Container 是树枝。
  • 操作系统的资源管理器
  • GUI中的容器层次图
  • XML、HTML文件解析
  • OA系统中,组织结构的处理
  • Junit单元测试框架:底层设计就是典型的组合模式,TestCase(叶子)、TestUnite(容器)、Test接口(抽象)

注意:别将组合模式与继承中的组合混淆

实现

使用组合模式模拟杀毒软件架构设计
步骤1、2:创建部分-整体需要实现的统一接口,然后创建具体的叶子组件(部分)和容器组件(整体)

package com.ly.composite;

import java.util.ArrayList;
import java.util.List;

/**
 * 使用组合模式模拟杀毒软件架构设计
 * liyang 2020-07-20
 */

public interface AbstractFile {
    void killVirus();  //杀毒
}

//叶子构件1-图片文件
class ImageFile implements AbstractFile {
    private String name;
    public ImageFile(String name) {
        this.name = name;
    }
    @Override
    public void killVirus() {
        System.out.println("图片文件杀毒:" + name + "查杀中...");
    }
}

//叶子构件2-文本文件
class TextFile implements AbstractFile {
    private String name;
    public TextFile(String name) {
        this.name = name;
    }
    @Override
    public void killVirus() {
        System.out.println("文本文件杀毒:" + name + "查杀中...");
    }
}

//叶子构件3-视频文件
class VideoFile implements AbstractFile {
    private String name;
    public VideoFile(String name) {
        this.name = name;
    }
    @Override
    public void killVirus() {
        System.out.println("视频文件杀毒:" + name + "查杀中...");
    }
}

//容器构件-文件夹
class Folder implements AbstractFile {
    private String name;
    //容器下的子节点,可能是一个容器构件,也可能是一个叶子构件
    private List<AbstractFile> list = new ArrayList<>();

    public Folder(String name) {
        this.name = name;
    }

    @Override
    public void killVirus() {
        System.out.println("文件夹杀毒:" + name + "查杀中...");
        for(AbstractFile file : list ) {
            file.killVirus();
        }
    }

    public void add(AbstractFile file) {
        list.add(file);
    }

    public void remove(AbstractFile file) {
        list.remove(file);
    }

    public AbstractFile getChild(int index) {
        if(index < list.size()) return list.get(index);
        return null;
    }
}

步骤3:客户端测试

package com.ly.composite;

/**
 * liyang 2020-07-20
 * 组合模式的客户端测试
 * 模式杀毒过程,组合模式使得对文件和文件夹的删除方式一致(从客户端角度来看,没有区别)
 */

public class Client {
    public static void main(String[] args) {
        //*************************************************************************************
        //AbstractFile中没有add方法,只有killVirus方法,因此按如下方法声明定义,需要在使用add方法时进行强转
        //AbstractFile file01 = new Folder("我的收藏");
        //((Folder) file01).add(file02);
        //*************************************************************************************

        Folder file01 = new Folder("我的收藏");
        Folder file02 = new Folder("Java文件夹");
        Folder file03 = new Folder("C++文件夹");
        file01.add(file02);
        file01.add(file03);
        //file01.killVirus();

        AbstractFile file04 = new VideoFile("Java视频");
        AbstractFile file05 = new VideoFile("C++视频");
        file02.add(file04);
        file03.add(file05);
        //file04.killVirus();

        AbstractFile file06 = new ImageFile("Java垃圾回收机制");
        AbstractFile file07 = new ImageFile("C++指针原理");
        file02.add(file06);
        file03.add(file07);

        AbstractFile file08 = new TextFile("Core Java Volume I: Fundamentals.pdf");
        AbstractFile file09 = new TextFile("C++ Primer.pdf");
        file02.add(file08);
        file03.add(file09);

        file01.killVirus();
    }
}

结果:

文件夹杀毒:我的收藏查杀中...
文件夹杀毒:Java文件夹查杀中...
视频文件杀毒:Java视频查杀中...
图片文件杀毒:Java垃圾回收机制查杀中...
文本文件杀毒:Core Java Volume I: Fundamentals.pdf查杀中...
文件夹杀毒:C++文件夹查杀中...
视频文件杀毒:C++视频查杀中...
图片文件杀毒:C++指针原理查杀中...
文本文件杀毒:C++ Primer.pdf查杀中...

Process finished with exit code 0

参考资料:

https://www.runoob.com/design-pattern/composite-pattern.html

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java实现多个设计模式的组合是非常常见的,在这里我将以组合模式和装饰器模式为例进行介绍。 首先,组合模式是一种结构型模式,它将对象组合成树形结构以表示“部分-整体”的层次结构,使得用户对单个对象和组合对象的使用具有一致性。在实现组合模式时,我们需要定义一个抽象的组件(Component)类,它可以是一个抽象类或接口,并且包含了一些公共的方法和属性。然后,我们定义一个叶子节点(Leaf)类,它实现了组件(Component)类的方法。最后,我们定义一个组合节点(Composite)类,它也实现了组件(Component)类的方法,并且包含了一个子节点列表,用于存储其子节点。这样,我们就可以通过组合节点(Composite)来表示整个树形结构。 其次,装饰器模式是一种结构型模式,它允许我们动态地给一个对象添加一些额外的职责,而不需要修改其原始类。在实现装饰器模式时,我们需要定义一个抽象的组件(Component)类,它可以是一个抽象类或接口,并且包含了一些公共的方法和属性。然后,我们定义一个具体的组件(ConcreteComponent)类,它实现了组件(Component)类的方法。最后,我们定义一个装饰器(Decorator)类,它也实现了组件(Component)类的方法,并且包含了一个指向组件(Component)的引用。这样,我们就可以通过装饰器(Decorator)来动态地为组件(Component)添加额外的职责。 为了实现多个设计模式的组合,在 Java 中,我们可以通过将装饰器(Decorator)类作为组合节点(Composite)类的子类来实现。具体地,我们可以定义一个装饰器组合节点(DecoratorComposite)类,它继承了组合节点(Composite)类,并且包含了一个指向装饰器(Decorator)的引用。这样,我们就可以通过装饰器组合节点(DecoratorComposite)来同时实现组合模式和装饰器模式的功能。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值