【设计模式】外观模式(Facade Pattern)

news2024/11/14 20:37:49

外观模式属于结构型模式,主要解决客户程序访问复杂程序中的多个子程序而产生的高耦合度高复杂度问题,根本目的在于简化接口的调用。例如我们去医院看病,可能要去挂号、门诊、划价、取药(子系统角色),这让患者或患者家属(客户程序)觉得很复杂,如果有提供接待人员(外观角色),只让接待人员来处理,就很方便。


文章目录

  • 外观模式的介绍
    • 优点
    • 缺点
    • 应用场景
    • 注意事项
  • 外观模式的使用
    • 类图
      • 两种角色
    • 实现方法
      • 第一步,编写子系统
      • 第二步,编写外观类
      • 第三步,编写测试类测试
    • 未使用外观模式的结构
      • 修改测试类测试
    • 本文参考



外观模式的介绍

外观模式也被称作门面模式,作用是将客户程序与一组子系统进行解耦,使得客户程序只需要与一个外观类打交道,而不需要与多个子系统类打交道。原理是定义一个单一类(外观类)来处理客户程序需要调用的多个功能类(子系统类),此时客户程序只需要调用该单一类即可完成对多个子系统的调用。

优点

  • 将程序程序与子系统分隔,减少系统的互相依赖,降低了系统的耦合性
  • 系统的修改对其他子系统没有影响,而且子系统内部变化也不会影响外观对象,提高了灵活性
  • 操作都在外观类中封装完毕,提高了调用的安全性
  • 唯一入口,只提供了一个访问子系统的唯一入口,但不会影响客户程序直接使用子系统类

缺点

  • 不符合开闭原则,如果设计不当,增加新的子系统可能需要修改外观类,继承重写都不合适

  • 不能限制客户程序使用子系统,外观模式不能很好地限制客户程序直接使用子系统,如果客户程序对访问子系统做太多的限制就会减少可变性与灵活性

应用场景

  • 当要为访问一系列复杂的子系统提供一个简单的入口时
  • 层次化结构中,可以使用外观模式定义系统中每一层的入口,层与层之间不直接产生联系,而通过外观类建立联系,降低层之间的耦合度,其中三层架构就是这样的一个例子

注意事项

  • 外观单例:很多情况下为了节约系统资源,系统只需要一个外观类的实例,也就是外观类可以是一个单例类,这样可以降低系统资源的消耗
  • 多个外观类:客户程序可以针对抽象外观类进行编程,在抽象外观类下可以设计多个外观子类,每个外观类负责和一些特定子对象交互,向客户程序提供相应业务功能
  • 不要通过外观类增加新行为:外观模式的意图是为子系统提供一个集中简化的沟通渠道,而不是向子系统中增加新行为,新行为的增加应该通过修改原有子系统类或增加新的子系统类来实现而不是通过外观类实现



外观模式的使用

​ 举例,每天下班回家的我还需要收拾家务,如扫地、拖地、抹尘、烧水等(子系统类),这让我不胜其烦,于是我决定开发一个机器人来帮我解决这些琐事,以后回家,点击机器人的“收拾家务”按钮就能让它自动收拾家务,岂不快哉。


类图

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nt2tEZW9-1670332394024)(../../../MDimages/%E5%A4%96%E8%A7%82%E6%A8%A1%E5%BC%8F_images/image-20221205203050059.png)]


两种角色

  • 外观角色(Facade):此角色中可以知道相关的一个或多个子系统的功能和责任,正常情况下将来自客户程序的请求委派到对应的子系统中去,传递给相应的子系统对象处理。
  • 子系统角色(SubSystem):每一个子系统可以是一个单独的类,也可以是一个功能模块。每一个子系统都可以被客户程序直接调用,或者被外观角色调用。子系统并不知道外观类的存在,对于子系统而已,外观角色仅仅是另一个客户程序。



实现方法


第一步,编写子系统

扫地功能类

package 设计模式.结构型模式.外观模式;

public class 扫地功能类 {
    public void 开始(){
        System.out.println("开始扫地……扫地完成!");
    }
}

拖地功能类

package 设计模式.结构型模式.外观模式;

public class 拖地功能类 {
    public void 开始(){
        System.out.println("开始拖地……拖地完成!");
    }
}

抹尘功能类

package 设计模式.结构型模式.外观模式;

public class 抹尘功能类 {
    public void 开始(){
        System.out.println("开始抹尘……抹尘完成!");
    }
}

烧水功能类

package 设计模式.结构型模式.外观模式;

public class 烧水功能类 {
    public void 开始(){
        System.out.println("开始烧水……水还有一阵子才开,先去干其他事吧!");
		// 延时操作
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000); // 等待一秒
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("水开了!");
            }
        }).start();
    }
}

第二步,编写外观类

机器人类

package 设计模式.结构型模式.外观模式;

/**
 * 外观角色(Facade)
 * 客户程序自己委托该类来调用一系列子功能
 */
public class 机器人类 {

    // 客户程序通过调用该方法来间接的与一组子功能通信
    public void 收拾家务(){
        // 先把水烧上
        new 烧水功能类().开始();
        // 再扫地
        new 扫地功能类().开始();
        // 然后拖地
        new 拖地功能类().开始();
        // 最后抹尘
        new 抹尘功能类().开始();
    }
}

第三步,编写测试类测试

测试类

package 设计模式.结构型模式.外观模式;

public class 测试类 {
    public static void main(String[] args) {
        // 回到家看到我的机器人
        机器人类 机器人 = new 机器人类();
        // 点击“收拾家务”按钮
        机器人.收拾家务();
    }
}

测试结果

开始烧水……水还有一阵子才开,先去干其他事吧!
开始扫地……扫地完成!
开始拖地……拖地完成!
开始抹尘……抹尘完成!
水开了!
Process finished with exit code 0

在这里插入图片描述



未使用外观模式的结构

不使用外观类,客户程序直接调用各个功能

修改测试类测试

测试类

package 设计模式.结构型模式.外观模式;

public class 测试类 {
    public static void main(String[] args) {
        // 回家先把水烧上
        new 烧水功能类().开始();
        // 再扫地
        new 扫地功能类().开始();
        // 然后拖地
        new 拖地功能类().开始();
        // 最后抹尘
        new 抹尘功能类().开始();
    }
}

测试结果

在这里插入图片描述



本文参考

设计模式学习笔记(十三):外观模式(知乎)

设计模式详解——外观模式(知乎)

外观模式(菜鸟教程)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/67360.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Redis主从复制、哨兵模式、集群模式

文章目录一、Redis主从复制主从复制流程Redis主从复制的作用二、哨兵模式1、哨兵模式集群架构2、哨兵模式主要功能3、哨兵监控整个系统节点的过程4、主观下线5、客观下线6、master 选举7、故障迁移8、优点与缺点三、Cluster群集1、集群的作用2、Redis集群的数据分片四、实验一&…

Linux||使用vi编辑器按上下键输出字母

一、问题描述 在终端执行sudo命令时,系统提示“vboxuser is not in the sudoers file.This incident will be reported”,因此需要修改/etc/sudoers文件。 使用Linux自带的vi编辑器修改/etc/sudoers文件时候,按i键进入编辑模式。使用键盘上下…

SD-Branch多分支组网解决方案

一、多分支组网建设背景 多分支一般是指行政上由总部和分支单位组成,总部和各分支处在地域不同的区域;由于业务访问需要,组网要求“总部-分支”互联打通。多分支网络组网方案主要针对的是总分企业、金融网点、连锁酒店、连锁门店、商超地产、…

组合预测 | MATLAB实现基于BP-Adaboost强分类器多特征分类预测

组合预测 | MATLAB实现基于BP-Adaboost强分类器多特征分类预测 目录 组合预测 | MATLAB实现基于BP-Adaboost强分类器多特征分类预测预测效果基本介绍模型特性程序设计参考资料预测效果 基本介绍 MATLAB实现基于BP-Adaboost强分类器多特征分类预测。在集成学习-Boosting,Bagging…

如何建立公司网站?【公司网站建设】

现在很多实体公司企业都会做一个自己的公司网站,毕竟一个网站等同于一张在互联网中的名片,可以让更多客户了解到你。那么如何建立公司网站?流程步骤大概是什么?今天给大家简单讲解一下。 一、注册域名 这个域名是就是网址&#…

学编程:Python入门考级必备[8]

目录 知识模块(8) 列表知识——井然有序 1.列表的创建 2.列表访问(查找操作) 3.列表增加操作 4.列表删除操作 5.统计与计算 附件代码: 炼 知识模块(8) 列表知识——井然有序 知识模块(8) 列表知识——井然有序 1.列表的创建 1.1 空列表: lie …

[Linux](16)网络编程:网络概述,网络基本原理,套接字,UDP,TCP,并发服务器编程,守护(精灵)进程

文章目录网络协议初识OSI 七层模型TCP/IP 四层(或五层)模型IP、MAC、端口号TCP 协议与 UDP 协议套接字套接字地址结构socket 函数bind 函数recvfrom 函数sendto 函数UDP 通信实现服务端客户端TCP 通信实现服务端listen 函数accept 函数实现客户端connect 函数实现改进&#xff…

项目中集成高德地图

如果文章对你有帮助欢迎【关注❤️❤️❤️点赞👍👍👍收藏⭐⭐⭐】一键三连!一起努力! 本文我们主要讲述的是怎么在自己的项目中集成高德地图。 一、注册账号并申请Key 首先我们打开浏览器,直接百度&…

python小知识--创建scrapy工程步骤

python小知识--创建scrapy工程步骤前言python中的scrapy框架是我们在平时使用爬虫使用比较多的框架首先第一步就是创建scrapy工程,下面通过pycharm软件来演示效果图1、右击文件名称选择open in terminal打开终端cmd2、在终端控制台输入scrapy startproject kuaikank…

grafana alert告警面板配置说明

grafana alert告警面板配置说明 前提:只有graph panel 也就是图表面板(一般都是折线图和柱状图或者点状图)可以添加Alert ,其他面板不支持。 Rule Name :Alert规则 的名称 每多少时间评估一次 如果配置了这个参数&…

安卓APP源码和设计报告——基于Android的垃圾分类系统

《移动应用开发》大作业报告 题 目基于Android的垃圾分类系统系 部班 级学 生 姓 名学 号指 导 教 师时 间 1、项目名称 垃圾分类系统 2、项目概述 近些年,由于人民生活水平是的提高,生活方式与生活节奏的加快,使我国的垃圾生产数量已远超…

Flink系列之Flink之Time和WaterMark深入剖析

title: Flink系列 一、Flink Window 常见需求背景 1.0 理论描述 需求描述: 每隔 5 秒,计算最近 10 秒单词出现的次数。 滑动窗口 每隔 5 秒,计算最近 5 秒单词出现的次数。 滚动窗口 第一个: 关于 TimeCharacteristic Process…

[附源码]Python计算机毕业设计Django医疗纠纷处理系统

项目运行 环境配置: Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术: django python Vue 等等组成,B/S模式 pychram管理等等。 环境需要 1.运行环境:最好是python3.7.7,…

[附源码]Python计算机毕业设计Django校园一卡通服务平台

项目运行 环境配置: Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术: django python Vue 等等组成,B/S模式 pychram管理等等。 环境需要 1.运行环境:最好是python3.7.7,…

HarmonyOS/OpenHarmony应用开发-FA卡片开发体验

卡片概述 卡片是一种界面展示形式,可以将应用的重要信息或操作前置到卡片,以达到服务直达,减少体验层级的目的。 卡片常用于嵌入到其他应用(当前只支持系统应用)中作为其界面的一部分显示,并支持拉起页面…

[附源码]Python计算机毕业设计Django校园疫情管理系统

项目运行 环境配置: Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术: django python Vue 等等组成,B/S模式 pychram管理等等。 环境需要 1.运行环境:最好是python3.7.7,…

优品汇系统开发机制介绍

优品汇系统通过通过消费增值模式,促进商品流通,打造中g最大的供应链。作为对政策的回应,绿点刺激实体经济。前期通过科学合理的业务体系,将大部分利润分配给客户和朋友,从而快速创造人气和粉丝数据。中期将逐步完善产品…

计算机网络学习笔记(Ⅲ):数据链路层

目录 1 数据链路层概述 1.1 基本概念 1.2 主要功能 2 封装成帧和透明传输 2.1 封装成帧 2.2 透明传输 1.字符计数法 2.字符填充法 3.零比特填充法 4.违规编码法 3 差错控制 3.1 差错 3.1 检错编码 1.奇偶校验码 2.CRC循环冗余码 3.2 纠错编码 1.确定校验码位数…

基于逆滤波算法的无约束图像超分辨重构研究-附Matlab代码

⭕⭕ 目 录 ⭕⭕✳️ 一、引言✳️ 二、逆滤波复原理论✳️ 三、实验验证✳️ 四、Matlab程序获取与验证✳️ 一、引言 图像复原( Image Restoration),也称图像恢复,是图像处理的一个重要方面。其目的就是尽可能地减少或去除在获取数字图像过程中发生的…

redis高级

redis持久化的两种方式:(重点) RDB: 全量打包-----------将内存中的所有数据存储在磁盘 执行一个bgsave ----1. 关机redis 2. 缺点:1.大量的数据重复打包和覆盖耗费时间和性能 2.配置save 不能把所有情况考虑在内,red…