设计模式——外观模式(Facade)

devtools/2024/9/23 4:27:01/

外观模式(Facade Pattern) 是一种结构型设计模式,它为一个子系统中的一组接口提供一个统一的高层接口,使得子系统更加容易使用。这种类型的设计模式属于结构型模式,它向客户端提供了一个接口,隐藏了子系统的复杂性。

1. 定义与特点

  • 定义外观模式为多个复杂的子系统提供一个对外的接口,使这些子系统更加容易的被访问。该模式对外有一个统一的接口,外部应用不用关心子系统内部的细节,大大降低了应用程序的复杂度,提高了可维护性。
  • 特点
    • 简化了调用过程,应用无需深入了解子系统。
    • 减少系统依赖,松散耦合。
    • 更好的划分访问层次。
    • 符合迪米特法则(最少知道原则)。

然而,外观模式也有一些缺点,比如增加子系统或扩展子系统行为容易引入风险,以及不符合开闭原则(即对扩展开放,对修改关闭)。

2. 角色

  • 外观(Facade)角色:这是模式的核心,客户端可以调用这个角色的方法。外观类知晓相关的(一个或者多个)子系统的功能和责任。在正常情况下,本角色会将所有从客户端发来的请求委派到相应的子系统去。
  • 子系统(Subsystem)角色:可以同时有一个或者多个子系统。每一个子系统都不是一个单独的类,而是一个类的集合。每一个子系统都可以被客户端直接调用,或者被外观角色调用。子系统并不知道外观的存在,对于子系统而言,外观仅仅是另外一个客户端而已。
  • 用户(Client)角色:用户通过外观类调用子系统的功能。

3. 适用场景

  • 子系统复杂:当子系统越来越复杂,增加外观模式可以提供简单调用接口。
  • 构建多层系统结构:利用外观对象作为每层的入口,简化层级调用。

4. 优缺点

  • 优点
    • 降低了子系统与客户端之间的耦合度,使得子系统内部的变化不会影响客户端。
    • 隐藏了子系统的复杂性,客户端只需要与外观类进行交互,无需了解子系统内部的实现细节。
    • 提高了系统的灵活性和可扩展性,通过修改外观类,可以很容易地改变子系统的行为。
  • 缺点
    • 在不恰当的情况下使用外观模式,可能会增加系统的复杂性,因为需要创建额外的外观类来管理子系统的接口。
    • 如果外观类过度使用,可能会导致它成为一个“上帝类”,该类知道太多的子系统细节,并承担过多的责任。这可能会使得系统的维护和扩展变得困难。

5. 示例

以下是一个简单的Java代码示例,演示了外观模式(Facade Pattern)的应用:

// 子系统A
class SubSystemA {public void operationA() {System.out.println("Subsystem A operationA() called.");}
}// 子系统B
class SubSystemB {public void operationB() {System.out.println("Subsystem B operationB() called.");}
}// 子系统C
class SubSystemC {public void operationC() {System.out.println("Subsystem C operationC() called.");}
}// 外观类
class Facade {private SubSystemA subSystemA;private SubSystemB subSystemB;private SubSystemC subSystemC;public Facade() {subSystemA = new SubSystemA();subSystemB = new SubSystemB();subSystemC = new SubSystemC();}// 提供给客户端调用的简单接口public void operation() {subSystemA.operationA();subSystemB.operationB();subSystemC.operationC();}
}// 客户端代码
public class Client {public static void main(String[] args) {// 使用外观类Facade facade = new Facade();facade.operation(); // 客户端只需要调用外观类的方法}
}

在这个示例中,我们有三个子系统类:SubSystemASubSystemBSubSystemC。每个子系统类都有一个操作方法(operationA()operationB()operationC())。然后,我们创建了一个外观类 Facade,该类聚合了这些子系统对象,并提供了一个 operation() 方法供客户端调用。在这个 operation() 方法中,我们按顺序调用了子系统的各个方法。

在客户端代码中,我们只需要创建一个 Facade 对象,并调用其 operation() 方法,而无需关心子系统内部的实现细节。这样,客户端与子系统之间的耦合度就降低了,同时提高了系统的灵活性和可扩展性。


http://www.ppmy.cn/devtools/39686.html

相关文章

ego_planner学习

飞行类型 enum TARGET_TYPE { MANUAL_TARGET 1, PRESET_TARGET 2, REFENCE_PATH 3 }; target_type_ TARGET_TYPE::PRESET_TARGET) trigger_sub_ nh.subscribe("/traj_start_trigger", 1, &EGOReplanFSM::triggerCallback, this); nh.param("fsm/waypo…

邂逅Linux--常见指令,万物为文件(一)

引子:在之前,我们经常听到Linux,那什么是Linux呢?Linux是一种免费使用和自由传播的类UNIX操作系统,其内核由林纳斯本纳第克特托瓦兹(Linus Benedict Torvalds)于1991年10月5日首次发布&#xff…

FFmpeg 中 -f 命令参数详解

FFmpeg FFmpeg是一个开源的、功能强大的多媒体框架,它能够处理几乎所有格式的音频和视频文件。FFmpeg由Fabrice Bellard创立,并由Michael Niedermayer等人继续开发。它包括了libavcodec(用于编解码)、libavformat(用于格式转换)、libavfilter(用于音视频过滤)、libavd…

探索大语言模型在信息提取中的应用与前景

随着人工智能技术的快速发展,大语言模型(LLMs)在自然语言处理(NLP)领域取得了显著的进展。特别是在信息提取(IE)任务中,LLMs展现出了前所未有的潜力和优势。信息提取是从非结构化文本…

小米poco x3 pro adbd 以root权限启动提供服务

1. 首先申请解锁 2. 按volume - 和 power开机进入fastboot 3.下载twrp-3.7.1_12-0-vayu.img 4. fastboot flash recoery twrp-3.7.1_12-0-vayu.img, 安装magisk, termux 5. 下载aosp的android11的aosp_arm64-img-11718355.zip,Artifact Viewer 6. 解压aosp_arm64-img-1…

Rust学习笔记(上)

前言 笔记的内容主要参考与《Rust 程序设计语言》,一些也参考了《通过例子学 Rust》和《Rust语言圣经》。 Rust学习笔记分为上中下,其它两个地址在Rust学习笔记(中)和Rust学习笔记(下)。 编译与运行 Ru…

SpringBoot+Redission实现排行榜功能

SpringBootRedission实现排行榜功能 demo地址:ranking-demo: 排行榜DEMO (gitee.com) 一、业务需求 实现一个排行榜,要求按照分数和达成这个分数的时间排序,即相同分数下,时间早的在上面 二、Redis中的zSet(有序集合) 1.简介 …

外包干了6天,技术明显进步

先说一下自己的情况,本科生,2019年我通过校招踏入了南京一家软件公司,开始了我的职业生涯。那时的我,满怀热血和憧憬,期待着在这个行业中闯出一片天地。然而,随着时间的推移,我发现自己逐渐陷入…