Java设计模式-组合模式(13)

简介: Java设计模式-组合模式(13)

大家好,我是馆长!今天开始我们讲的是结构型模式中的组合模式。老规矩,讲解之前再次熟悉下结构型模式包含:代理模式、适配器模式、桥接模式、装饰器模式、外观模式、享元模式、组合模式,共7种设计模式。

组合模式(Composite Pattern)

定义

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

解决问题

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

实现

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

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

结构

主要角色:

抽象构件(Component)角色:它的主要作用是为树叶构件和树枝构件声明公共接口,并实现它们的默认行为。在透明式的组合模式中抽象构件还声明访问和管理子类的接口;在安全式的组合模式中不声明访问和管理子类的接口,管理工作由树枝构件完成。

树叶构件(Leaf)角色:是组合中的叶节点对象,它没有子节点,用于实现抽象构件角色中 声明的公共接口。

树枝构件(Composite)角色:是组合中的分支节点对象,它有子节点。它实现了抽象构件角色中声明的接口,它的主要作用是存储和管理子部件,通常包含 add()、remove()、getSubList() 等方法。

应用场景:

在需要表示一个对象整体与部分的层次结构的场合。

要求对用户隐藏组合对象与单个对象的不同,用户可以用统一的接口使用组合结构中的所有对象的场合。

注意:

1.抽象构件可以定义成抽象类或者接口。

2.具体的实现可在树枝构件和树叶构件子类实现不同的功能。

3.可层层扩展树枝构件和树叶构件的子类。

优点:

组合模式使得客户端代码可以一致地处理单个对象和组合对象,无须关心自己处理的是单个对象,还是组合对象,这简化了客户端代码;

更容易在组合体内加入新的对象,客户端不会因为加入了新的对象而更改源代码,满足“开闭原则”;

缺点:

1.设计较复杂,客户端需要花更多时间理清类之间的层次关系;

2.不容易限制容器中的构件;

3.不容易用继承的方法来增加构件的新功能;

代码实现:文件夹管理

// 抽象构件 Resource
public abstract class Resource {
protected Integer id;
protected String name;

Resource(Integer id, String name) {
    this.id = id;
    this.name = name;
}

Resource() {

}

}

​​​​​​​
//树枝构件 FolderResource
@Data
public class FolderResource extends Resource {

FolderResource(Integer id,String name){
    super(id,name);
}


private List<Resource> children = new ArrayList<>();


public void remove(Resource resource) {
    children.remove(resource);
}

public void add(Resource resource) {
    this.children.add(resource);
}

public List<Resource> getChildren() {
    return this.children;
}

}
​​​​​​​
//树叶构件 FileResource
@Data
public class FileResource extends Resource{
FileResource() {
super();
}
FileResource(Integer id, String name) {
super(id, name);
}

@Override
public String toString() {
    return "FileResource{" +
            "id=" + id +
            "name=" + name +
            '}';
}

}

//模拟客户端 ClientDemo
public class ClientDemo {
public static void main(String[] args) {
FolderResource top = new FolderResource(1, "顶级目录");
FileResource file1 = new FileResource(4, "文件1.doc");
FileResource file2 = new FileResource(5, "文件2.doc");
FolderResource top1 = new FolderResource(2, "二级目录1");
FileResource file3 = new FileResource(6, "文件3.doc");
FileResource file4 = new FileResource(7, "文件4.doc");

    FolderResource top2 = new FolderResource(3, "二级目录2");
    FileResource file5 = new FileResource(8, "文件5.doc");
    FileResource file6 = new FileResource(9, "文件6.doc");

    top1.add(file3);
    top1.add(file4);

    top2.add(file5);
    top2.add(file6);

    top.add(top1);
    top.add(top2);

    top.add(file1);
    top.add(file2);
    Integer level = 0;
    printFolder(top, level);

}

public static void printFolder(FolderResource folder, Integer level) {
    String space = "";
    for (int i = 0; i < level; i++) {
        space += "  ";
    }
    System.out.println(space + folder.name);
    for (Resource item : folder.getChildren()) {
        if (item instanceof FolderResource) {
            Integer l = level + 1;
            printFolder((FolderResource) item, l);
        } else if (item instanceof FileResource) {
            String p = space + "  ";
            System.out.println(p + item.name);
        }
    }
}

}

扩展

在实际使用过程中,如 Java AWT/Swing 中的简单组件 JTextComponent 有子类 JTextField、JTextArea,容器组件 Container 也有子类 Window、Panel。再如:部分、整体场景,如树形菜单,文件、文件夹的管理。

好了,关于组合模式的说明,馆长就先讲到这里。谢谢各位看官!!

23 种设计模式不是孤立存在的,很多模式之间存在一定的关联关系,在大的系统开发中常常同时使用多种设计模式,或者模式与模式之间的组合进行生成更加强大的程序功能。

相关文章
|
1月前
|
设计模式 算法 Java
Java中的设计模式:提升代码质量的秘诀
【8月更文挑战第23天】在Java开发中,设计模式是提高代码可读性、可维护性和扩展性的强有力工具。本文通过浅显易懂的语言和实际案例,探讨几种常见的设计模式及其在Java中的应用,旨在帮助开发者更好地理解并运用这些模式来优化自己的代码结构。
46 2
|
13天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
155 37
|
9天前
|
设计模式 Java
Java设计模式:组合模式的介绍及代码演示
组合模式是一种结构型设计模式,用于将多个对象组织成树形结构,并统一处理所有对象。例如,统计公司总人数时,可先统计各部门人数再求和。该模式包括一个通用接口、表示节点的类及其实现类。通过树形结构和节点的通用方法,组合模式使程序更易扩展和维护。
Java设计模式:组合模式的介绍及代码演示
|
13天前
|
设计模式 安全 算法
【Java面试题汇总】设计模式篇(2023版)
谈谈你对设计模式的理解、七大原则、单例模式、工厂模式、代理模式、模板模式、观察者模式、JDK中用到的设计模式、Spring中用到的设计模式
【Java面试题汇总】设计模式篇(2023版)
|
13天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑】设计模式——原型模式
对比原型模式和传统方式的实现思路、代码方案、优缺点,阐述原型模式的使用场景,以及深拷贝、浅拷贝等相关概念,并扩展原型模式在Spring源码中的应用。
【Java笔记+踩坑】设计模式——原型模式
|
4天前
|
设计模式 安全 Java
Java 编程中的设计模式:单例模式的深度解析
【9月更文挑战第22天】在Java的世界里,单例模式就像是一位老练的舞者,轻盈地穿梭在对象创建的舞台上。它确保了一个类仅有一个实例,并提供全局访问点。这不仅仅是代码优雅的体现,更是资源管理的高手。我们将一起探索单例模式的奥秘,从基础实现到高级应用,再到它与现代Java版本的舞蹈,让我们揭开单例模式的面纱,一探究竟。
22 11
|
28天前
|
设计模式 缓存 算法
揭秘策略模式:如何用Java设计模式轻松切换算法?
【8月更文挑战第30天】设计模式是解决软件开发中特定问题的可重用方案。其中,策略模式是一种常用的行为型模式,允许在运行时选择算法行为。它通过定义一系列可互换的算法来封装具体的实现,使算法的变化与客户端分离。例如,在电商系统中,可以通过定义 `DiscountStrategy` 接口和多种折扣策略类(如 `FidelityDiscount`、`BulkDiscount` 和 `NoDiscount`),在运行时动态切换不同的折扣逻辑。这样,`ShoppingCart` 类无需关心具体折扣计算细节,只需设置不同的策略即可实现灵活的价格计算,符合开闭原则并提高代码的可维护性和扩展性。
39 2
|
28天前
|
设计模式 Java
Java 设计模式之谜:工厂模式与抽象工厂模式究竟隐藏着怎样的神奇力量?
【8月更文挑战第30天】在Java编程中,设计模式为常见问题提供了高效解决方案。工厂模式与抽象工厂模式是常用的对象创建型设计模式,能显著提升代码的灵活性、可维护性和可扩展性。工厂模式通过定义创建对象的接口让子类决定实例化哪个类;而抽象工厂模式则进一步提供了一个创建一系列相关或相互依赖对象的接口,无需指定具体类。这种方式使得系统更易于扩展和维护。
31 1
|
28天前
|
设计模式 Java
重构你的代码:探索Java中的混合、装饰器与组合设计模式
【8月更文挑战第30天】在软件开发中,设计模式为特定问题提供了结构化的解决方案,使代码更易理解、维护及扩展。本文将介绍三种常用的 Java 设计模式:混合模式、装饰器模式与组合模式,并附有示例代码展示实际应用。混合模式允许通过继承多个接口或抽象类实现多重继承;装饰器模式可在不改变对象结构的情况下动态添加新功能;组合模式则通过树形结构表示部分-整体层次,确保客户端处理单个对象与组合对象时具有一致性。
18 1
|
1月前
|
设计模式 算法 安全
Java编程中的设计模式:提升代码的可维护性和扩展性
【8月更文挑战第19天】在软件开发的世界里,设计模式是解决常见问题的一种优雅方式。本文将深入探讨Java编程语言中常用的几种设计模式,并解释如何通过这些模式来提高代码的可维护性和扩展性。文章不涉及具体的代码实现,而是侧重于理论和实践相结合的方式,为读者提供一种思考和改善现有项目的新视角。