软件设计模式系列之十二——外观模式

在软件设计中,经常会遇到需要与复杂子系统进行交互的情况。为了简化客户端与子系统之间的交互,提高系统的可维护性和可用性,外观模式应运而生。外观模式(Facade Pattern)是一种结构型设计模式,它提供一个统一的界面,用于访问系统中的一组相关接口,从而隐藏了系统的复杂性。在本文中,我们将深入探讨外观模式,包括其定义、举例说明、结构、实现步骤、代码实现、典型应用场景、优缺点、类似模式以及最后的小结。

1 模式的定义

外观模式是一种结构型设计模式,它提供了一个简化的接口,用于访问系统中的一组相关接口,以隐藏系统的复杂性。外观模式的主要目标是简化客户端与子系统之间的交互,同时降低了系统的耦合度。它允许客户端通过一个统一的入口点来与系统进行通信,而不需要了解系统内部的具体细节和复杂性。

2 举例说明

让我们通过几个简单的例子来说明外观模式的概念。
音响系统的例子。假设我们正在开发一个音响系统,该系统包括音响控制、CD播放器、收音机和音箱等组件。客户端希望能够简单地控制音响的各种功能,而不需要直接与每个组件进行交互。
在这里插入图片描述

智能手机的例子。智能手机的操作系统(如iOS和Android)为用户提供了一个外观,通过该外观可以轻松访问手机的各种功能,包括拨打电话、发送短信、浏览互联网、使用应用程序等。用户不需要了解手机的硬件和操作系统内部的复杂性。

家用电器的例子。一些现代家用电器,如洗衣机、洗碗机和微波炉,配备了控制面板,通过该面板用户可以选择不同的功能和设置。这些控制面板提供了一个简化的外观,使用户能够轻松操作家用电器。

汽车的驾驶控制台的例子。现代汽车的驾驶控制台包括了一系列按钮、开关和显示屏,通过这些控制界面,驾驶员可以控制汽车的各种功能,如调整座位、开启空调、切换收音机频道、导航、调整车速等。这些控制界面提供了一个简化的外观,使驾驶员能够方便地控制汽车。

3 结构

外观模式的结构包括以下主要组件:
在这里插入图片描述

外观(Facade):外观是外观模式的核心组件,它提供了一个简化的接口,用于与系统中的一组相关接口进行交互。外观负责委派请求给相应的子系统对象。

子系统(Subsystems):子系统是系统中的各个组件或模块,它们实现了系统的具体功能。外观通过与子系统协作来完成客户端的请求。

客户端(Client):客户端是使用外观模式的类或模块,它通过外观来简化与系统的交互。客户端不需要直接与子系统的具体类进行通信。

4 实现步骤

要实现外观模式,可以按照以下步骤进行操作:

定义子系统:首先,定义系统中的各个子系统,每个子系统负责实现一部分功能。

创建外观类:创建一个外观类,它包含对子系统的引用,并提供一个简化的接口,用于客户端访问系统的功能。

在外观类中委派请求:在外观类中实现方法,将客户端的请求委派给适当的子系统对象,以完成具体的操作。

客户端使用外观类:客户端通过外观类来访问系统的功能,而不需要直接与子系统的具体类进行交互。

5 代码实现

让我们使用Java代码来实现上面的音响系统的外观模式:

首先,我们创建外观类 StereoFacade:

public class StereoFacade {private StereoControl stereoControl;private CDPlayer cdPlayer;private Radio radio;private Speakers speakers;public StereoFacade() {stereoControl = new StereoControl();cdPlayer = new CDPlayer();radio = new Radio();speakers = new Speakers();}public void playMusic() {stereoControl.turnOn();cdPlayer.play();radio.tune();speakers.volumeUp();}public void turnOff() {stereoControl.turnOff();cdPlayer.stop();radio.turnOff();speakers.volumeDown();}
}

接下来,我们可以在客户端使用外观类:

public class ClientWithFacade {public static void main(String[] args) {StereoFacade stereoFacade = new StereoFacade();// 使用外观模式简化操作stereoFacade.playMusic();// 关闭音响stereoFacade.turnOff();}
}

通过外观模式,客户端只需与 `外观类 StereoFacade 交互,而不需要了解音响系统的具体子系统。这大大简化了客户端的代码,并提高了系统的可维护性和可用性。

6 典型应用场景

外观模式通常在以下情况下得到广泛应用:

简化复杂系统。当系统包含多个复杂的子系统或模块,并且客户端需要与这些子系统进行交互时,外观模式可以提供一个简化的接口,以减少客户端的复杂性。

解耦客户端与子系统。外观模式允许客户端与系统的具体实现解耦,使得系统的更改不会影响到客户端。

提供高层接口。外观模式可以为系统提供一个高层接口,隐藏底层组件的复杂性,使客户端更容易使用。

构建库或框架。在设计库或框架时,外观模式可以提供一个简单的接口,以便其他开发者能够轻松使用库中的功能。

7 优缺点

外观模式具有一些优点和缺点,让我们来看看:

优点:

简化接口:外观模式提供了一个简化的接口,使客户端更容易使用系统的功能。

降低耦合度:外观模式将客户端与子系统的具体实现解耦,允许系统的更改不会影响到客户端。

提高可维护性:由于外观模式隐藏了系统的复杂性,因此提高了系统的可维护性,减少了维护成本。

提供了更高层次的接口:外观模式允许为系统提供高层次的接口,有助于组织和管理复杂的代码。

缺点:

不符合开闭原则:如果系统的功能需要变化或扩展,可能需要修改外观类,这可能会违反开闭原则。

可能引入单点故障:外观模式将多个子系统封装在一个外观类中,如果外观类发生故障,整个系统可能会受到影响。

8 类似模式

与外观模式类似的模式包括以下几种,它们都涉及到简化复杂系统的接口或交互,但在目的和实现上略有不同。

适配器模式(Adapter Pattern)

适配器模式和外观模式都是结构型设计模式,它们都涉及到简化接口或交互,以便客户端能够更容易地使用系统的功能。适配器模式通常用于解决接口不兼容的问题,将一个接口转换为另一个接口。外观模式用于提供一个统一的界面,隐藏系统的复杂性,使客户端更容易使用。

代理模式(Proxy Pattern)

代理模式和外观模式都涉及到一个对象(代理或外观)充当客户端与系统之间的中介,以控制访问系统的功能。代理模式主要用于控制访问对象,通常包括延迟加载、访问控制或监控。外观模式主要用于简化客户端与系统的交互,隐藏系统的复杂性。

装饰模式(Decorator Pattern)

装饰模式和外观模式都是结构型设计模式,它们都涉及到对象的包装和功能扩展。装饰模式允许在运行时动态地添加功能,而不改变对象的接口。外观模式提供一个统一的接口,用于访问系统的一组相关接口,目的是隐藏系统的复杂性。

这些模式都与简化系统的接口或交互有关,但它们的重点和用途略有不同。在实际应用中,应根据具体问题和需求选择最合适的设计模式。

9 小结

外观模式是一种有助于简化复杂系统的结构型设计模式。它提供了一个统一的界面,用于访问系统中的一组相关接口,从而隐藏了系统的复杂性。通过外观模式,客户端可以更轻松地与系统进行交互,而不需要了解系统内部的具体细节。这种模式提高了系统的可维护性、可用性,并降低了客户端的复杂性。在设计和开发复杂系统时,外观模式可以成为一个有用的工具,以提高代码的可维护性和可扩展性。

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

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

相关文章

激活函数总结(四十一):激活函数补充(ShiLU、ReLUN)

激活函数总结(四十一):激活函数补充 1 引言2 激活函数2.1 ShiLU激活函数2.2 ReLUN激活函数 3. 总结 1 引言 在前面的文章中已经介绍了介绍了一系列激活函数 (Sigmoid、Tanh、ReLU、Leaky ReLU、PReLU、Swish、ELU、SELU、GELU、Softmax、Sof…

腾讯云16核服务器性能测评_轻量和CVM配置大全

腾讯云16核服务器配置大全,CVM云服务器可选择标准型S6、标准型SA3、计算型C6或标准型S5等,目前标准型S5云服务器有优惠活动,性价比高,计算型C6云服务器16核性能更高,轻量16核32G28M带宽优惠价3468元15个月,…

简单的手机电脑无线传输方案@固定android生成ftp的IP地址(android@windows)

文章目录 abstractwindows浏览android文件环境准备客户端软件无线网络链接步骤其他方法 手机浏览电脑文件公网局域网everythingpython http.server 高级:固定android设备IP准备检查模块是否生效 windows 访问ftp服务器快捷方式命令行方式双击启动方式普通快捷方式映射新的网络位…

第六次面试、第一次复试

第六面: hr迟到,说是搞错了以为线下,我打电话过去才开始,问我想电话面还是视频,果断电话面 自我介绍 介绍了一下公司的工作 ................. 项目拷打: grpc数据如何传输的如何调用两个接口如何获取…

Android 12 源码分析 —— 应用层 六(StatusBar的UI创建和初始化)

Android 12 源码分析 —— 应用层 六(StatusBar的UI创建和初始化) 在前面的文章中,我们分别介绍了Layout整体布局,以及StatusBar类的初始化.前者介绍了整体上面的布局,后者介绍了三大窗口的创建的入口处,以及需要做的准备工作.现在我们分别来细化三大窗口的UI创建和…

利用大模型知识图谱技术,告别繁重文案,实现非结构化数据高效管理

我,作为一名产品经理,对文案工作可以说是又爱又恨,爱的是文档作为嘴替,可以事事展开揉碎讲清道明;恨的是只有一个脑子一双手,想一边澄清需求一边推广宣传一边发布版本一边申报认证实在是分身乏术&#xff0…

模块化软件架构:使用单体、微服务和模块化单体的优缺点

【squids.cn】 全网zui低价RDS,免费的迁移工具DBMotion、数据库备份工具DBTwin、SQL开发工具等 近年来,微服务架构在大多数软件解决方案中已经处于领先地位,而且在很多情况下,它经常被选择为我们开始开发的架构。然而&#xff0c…

解密list的底层奥秘

🎈个人主页:🎈 :✨✨✨初阶牛✨✨✨ 🐻强烈推荐优质专栏: 🍔🍟🌯C的世界(持续更新中) 🐻推荐专栏1: 🍔🍟🌯C语言初阶 🐻推荐专栏2: 🍔…

软件定制APP开发步骤分析|小程序

软件定制APP开发步骤分析|小程序 软件定制开发步骤: 1.需求分析: 这是软件定制开发的第一步,也是最关键的一步。在这个阶段,软件开发团队需要与客户进行沟通,了解客户的具体需求和期望。通过讨论和交流,确…

前后端分离管理系统day01---Springboot+MybatisPlus

目录 目录 软件 基础知识 一创建后端项目 注意: 删除多余项 创建测试类 二 加入mybatis-plus依赖支持 1.加入依赖码 2.创建数据库实例/创建用户表/插入默认数据 创建数据库实例 创建表 插入数据 3.配置yml文件 注意:wms01必须是数据库的名字&…

[maven] scopes 管理 profile 测试覆盖率

[maven] scopes & 管理 & profile & 测试覆盖率 这里将一些其他的特性和测试覆盖率(主要是 jacoco) scopes maven 的 scope 主要就是用来限制和管理依赖的传递性,简单的说就是,每一个 scope 都有其对应的特性&…

WebGL 初始化着色器

目录 前言 初始化着色器的7个步骤 创建着色器对象(gl.createShader()) gl.createShader()规范 gl.deleteShader()规范 指定着色器对象的代码(gl.shaderSource&…

Linux高并发服务器开发第四章:Linux网络编程

1. 网络结构模式 C/S结构 简介 服务器 - 客户机,即 Client - Server(C/S)结构。C/S 结构通常采取两层结构。服务器负责数据的管理,客户机负责完成与用户的交互任务。客户机是因特网上访问别人信息的机器,服务器则是…

【结构型】代理模式(Proxy)

目录 代理模式(Proxy)适用场景代理模式实例代码(Java) 代理模式(Proxy) 为其他对象提供一种代理以控制对这个对象的访问。Proxy 模式适用于在需要比较通用和复杂的对象指针代替简单的指针的时候。 适用场景 远程代理 (Remote Proxy) 为一个对象在不同…

10分钟设置免费海外远程桌面

前言 本教程将向您介绍如何使用 Amazon Lightsail 服务的免费套餐轻松搭建属于您的远程桌面。依托于 Amazon 全球可用区,您可以在世界各地搭建符合您配置需求的远程桌面。 本教程需要先拥有亚马逊云科技海外账户。现在注册亚马逊云科技账户可以享受12个月免费套餐…

Python与数据分析--每天绘制Matplotlib库实例图片3张-第1天

目录 1.实例1--Bar color demo 2.实例2--Bar Label Demo 3.实例3--Grouped bar chart with labels 1.实例1--Bar color demo import matplotlib.pyplot as plt # 支持中文 plt.rcParams[font.sans-serif] [SimHei] # 用来正常显示中文标签 plt.rcParams[axes.unicode_minus…

Linux系统编程——网络编程的学习

Linux系统编程学习相关博文 Linux系统编程——文件编程的学习Linux系统编程——进程的学习Linux系统编程——进程间通信的学习Linux系统编程——线程的学习 Linux系统编程——网络编程的学习 一、概述1. TCP/UDP2. 端口号3. 字节序4. Sockt服务器和客户端的开发步骤1. 服务器2…

Vue路由与node.js环境搭建

目录 前言 一.Vue路由 1.什么是spa 1.1简介 1.2 spa的特点 1.3 spa的优势以及未来的挑战 2.路由的使用 2.1 导入JS依赖 2.2 定义两个组件 2.3 定义组件与路径对应关系 2.4 通过路由关系获取路由对象 2.5 将对象挂载到vue实例中 2.6 定义触发路由事件的按钮 2.7 定…

利用cms主题构造木马(CVE-2022-26965)

简介 CVE-2022-26965是Pluck CMS 4.7.16版本存在一个远程shell上传执行漏洞。 攻击者可利用此漏洞通过构造恶意的主题包进行上传并执行,未经授权访问服务器,造成潜在的安全隐患。 过程 1.打开环境,查看源码,发现login.php 2.进…

如何制作一个成功的超市购物小程序

随着互联网的普及和移动支付的便捷性,越来越多的消费者选择在网上购物,这也促使越来越多的商家开始搭建自己的小程序商城。对于超市便利店来说,拥有一个便捷、易用的小程序商城能够吸引更多的消费者,提高销售效率。那么如何快速搭…