Java二十三种设计模式-外观模式(9/23)

外观模式:简化复杂系统的统一接口

引言

外观模式(Facade Pattern)是一种结构型设计模式,它为子系统中的一组接口提供一个统一的高层接口。外观模式定义了一个可以与复杂子系统交互的简化接口,使得子系统更加易于使用。

基础知识,java设计模式总体来说设计模式分为三大类:

(1)创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

(2)结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式。

(3)行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

第一部分:外观模式概述

1.1 定义与用途

外观模式的基本定义

外观模式(Facade Pattern)是一种常用的结构型设计模式,其核心目的是为一个复杂的子系统提供一个简化的统一接口。通过外观模式,客户端可以访问一个复杂的类系统,而无需了解该系统内部的复杂性。

解释为何需要外观模式

  • 简化接口:在大型软件系统中,尤其是当系统由多个类和组件构成时,客户端代码可能需要与这些复杂的类交互。外观模式通过提供一个简化的接口,简化了客户端与这些类的交互。
  • 降低耦合度:外观模式降低了客户端与复杂子系统的耦合度,客户端不依赖于子系统的具体实现,只依赖于外观对象。
  • 易于维护和扩展:当子系统需要修改或扩展时,只要接口保持不变,客户端代码就不需要修改,从而提高了系统的可维护性和可扩展性。

1.2 外观模式的组成

外观(Facade)

  • 定义:外观是一个接口,它提供了一组特定的方法,用于访问子系统中的复杂功能。
  • 角色:作为客户端与子系统交互的中介,外观对象将客户端的请求委托给子系统中的相应对象。

子系统(Subsystem)

  • 定义:子系统是一系列类和接口的集合,它们实现了特定的功能。
  • 角色:子系统包含了业务逻辑的具体实现,外观对象通过子系统来完成请求的处理。

客户端(Client)

  • 角色:客户端使用外观对象来访问子系统的功能,客户端不直接与子系统交互。

外观模式通过引入外观对象,将客户端与复杂的子系统解耦,使得客户端可以更加容易地使用子系统的功能。在下一部分中,我们将通过Java代码示例来展示外观模式的具体实现。

第二部分:外观模式的实现

2.1 Java实现示例

以下是使用Java语言实现外观模式的代码示例。假设我们有一个复杂的家庭影院系统,包括多个组件如投影仪、音响和屏幕。

// 子系统接口
interface TheaterControl {void on();void off();void setVolume(int volume);
}// 具体子系统:投影仪
class Projector implements TheaterControl {public void on() { System.out.println("Projector on."); }public void off() { System.out.println("Projector off."); }// 投影仪没有音量设置
}// 具体子系统:音响
class StereoSystem implements TheaterControl {public void on() { System.out.println("Stereo on."); }public void off() { System.out.println("Stereo off."); }public void setVolume(int volume) {System.out.println("Stereo volume set to " + volume);}
}// 具体子系统:屏幕
class Screen implements TheaterControl {public void on() { System.out.println("Screen on."); }public void off() { System.out.println("Screen off."); }// 屏幕没有音量设置
}// 外观
class HomeTheaterFacade {private TheaterControl projector;private TheaterControl stereoSystem;private TheaterControl screen;public HomeTheaterFacade() {this.projector = new Projector();this.stereoSystem = new StereoSystem();this.screen = new Screen();}public void watchMovie(String movie) {System.out.println("Get ready to watch a " + movie + "!");turnOn();setVolume(5);projector.on();screen.on();}private void turnOn() {projector.on();stereoSystem.on();screen.on();}private void setVolume(int volume) {stereoSystem.setVolume(volume);}
}// 客户端代码
public class Client {public static void main(String[] args) {HomeTheaterFacade homeTheater = new HomeTheaterFacade();homeTheater.watchMovie("Java Design Patterns");}
}

2.2 外观模式中的角色和职责

外观(Facade)

  • 职责:为复杂的子系统提供一个简化的接口。外观对象通常包含多个子系统对象的引用,并将客户端的请求委托给相应的子系统对象。
  • 实现:外观类通常包含子系统的引用,并提供一系列方法,这些方法将复杂的操作封装成简单的步骤。

子系统(Subsystem)

  • 职责:实现特定的功能,子系统可以独立于外观和其他子系统运行。
  • 实现:每个子系统都是一个类,实现特定的功能,如上述示例中的ProjectorStereoSystemScreen

客户端(Client)

  • 职责:使用外观对象来访问子系统的功能,客户端不直接与子系统交互。
  • 实现:客户端通过外观对象的方法来执行操作,如上述示例中的Client类。

相互作用

  • 客户端调用:客户端调用外观对象的方法。
  • 外观委托:外观对象将请求委托给一个或多个子系统。
  • 子系统执行:子系统执行具体的操作并返回结果(如果有的话)。

外观模式通过引入外观对象,简化了客户端与复杂子系统的交互,降低了系统间的耦合度,提高了系统的可维护性和可扩展性。在下一部分中,我们将探讨外观模式的使用场景。

 

第三部分:外观模式的使用场景

3.1 简化复杂系统的访问

在软件开发中,经常会遇到一些复杂的系统,这些系统可能包含多个组件和子系统,每个部分都有其特定的功能和接口。对于客户端来说,直接与这些复杂的系统交互可能会非常困难和繁琐。

外观模式的应用:

  • 统一接口:外观模式提供一个统一的高层接口,使得客户端可以通过这个简化的接口与复杂系统交互,而无需了解内部的复杂性。
  • 简化客户端逻辑:通过外观模式,客户端的代码可以更加简洁和易于理解,因为客户端只需要与外观对象交互,而不是直接与复杂的子系统交互。
  • 易于使用:外观模式隐藏了复杂的实现细节,使得即使是不熟悉系统内部结构的用户也能容易地使用系统。

应用实例:

  • 操作系统的命令行界面:提供了一个简化的接口来执行复杂的系统命令。
  • 智能家居控制系统:通过一个简单的界面控制家中的多个智能设备,如灯光、温度、安全系统等。

3.2 降低系统间的耦合度

在大型软件项目中,系统通常由多个模块或服务组成,这些模块之间可能存在紧密的耦合关系。耦合度过高会导致系统难以维护和扩展。

外观模式的优势:

  • 解耦系统组件:外观模式通过引入一个外观层,将客户端与子系统的直接依赖关系转换为对外观的依赖,从而降低了系统间的耦合度。
  • 提高模块化:通过降低模块间的直接依赖,系统可以更加模块化,每个模块可以独立地开发和测试。
  • 易于扩展和维护:当系统需要扩展或修改时,只要外观接口保持不变,就不需要修改依赖于这个接口的客户端代码。

应用实例:

  • 支付系统集成:在电子商务平台中,可能需要集成多种支付方式,如信用卡、PayPal、微信支付等。通过外观模式,可以为这些支付方式提供一个统一的接口,降低它们之间的耦合度。
  • 中间件服务:在企业应用中,中间件服务如数据库连接池、消息队列等,可以通过外观模式提供简化的接口,使得应用程序更容易与这些服务交互。

外观模式通过提供一个简化的接口来简化复杂系统的访问,并降低系统间的耦合度。这使得系统更加易于使用、维护和扩展。在下一部分中,我们将讨论外观模式的优点与缺点。

第四部分:外观模式的优点与缺点

4.1 优点

简化接口

  • 统一访问点:为复杂的子系统提供一个简化的统一接口,简化了客户端的交互方式。

降低耦合度

  • 减少依赖:客户端不直接依赖于复杂的子系统,而是通过外观对象进行交互,降低了耦合度。

提高可维护性

  • 易于修改和扩展:当子系统需要修改或扩展时,只要外观接口保持不变,客户端代码不需要修改。

隐藏实现细节

  • 保护子系统:外观模式隐藏了子系统的实现细节,客户端不需要了解内部的复杂逻辑。

4.2 缺点

过度封装

  • 隐藏过度:如果外观对象封装过度,可能会隐藏必要的信息,使得客户端难以使用子系统的全部功能。

更新困难

  • 接口变更:一旦外观对象的接口被广泛使用,对其进行更新或修改可能会变得困难。

性能问题

  • 潜在的性能开销:在某些情况下,外观模式可能会引入额外的性能开销,尤其是在需要频繁调用多个子系统方法时。

第五部分:外观模式与其他模式的比较

5.1 与适配器模式的比较

适配器模式

  • 目的:使不兼容的接口能够一起工作,主要用于接口转换。
  • 实现:通常通过一个适配器类来转换接口。

外观模式

  • 目的:为复杂的子系统提供一个简化的统一接口,主要用于简化客户端与子系统的交互。

对比

  • 使用场景:适配器模式主要用于解决接口不兼容的问题,而外观模式主要用于简化对复杂子系统的访问。

5.2 与代理模式的对比

代理模式

  • 目的:为另一个对象提供一个代替或占位符,以控制对它的访问。
  • 实现:代理对象在客户端和真实对象之间起到中介的作用。

外观模式

  • 目的:为复杂的子系统提供一个简化的统一接口。

对比

  • 控制访问:代理模式主要用于控制对对象的访问,而外观模式主要用于简化对复杂子系统的访问。
  • 使用场景:代理模式适用于需要控制访问权限或延迟初始化的场景,外观模式适用于需要简化对复杂系统操作的场景。

外观模式通过提供一个简化的接口来简化复杂系统的访问,并降低系统间的耦合度。然而,合理使用外观模式并避免其缺点是至关重要的。了解其替代方案和与其他模式的比较可以帮助开发者根据具体需求和场景选择最合适的设计模式。在实际开发中,应根据具体情况灵活运用外观模式,以达到最佳的设计效果。

第六部分:外观模式的最佳实践和建议

6.1 最佳实践

保持外观的简洁性

  • 单一职责:确保外观类保持简洁,专注于提供一个简化的接口,而不是实现复杂的逻辑。

明确外观的职责

  • 职责分离:外观类应该只负责协调子系统的操作,不包含业务逻辑。

避免过度封装

  • 适度封装:避免隐藏过多的信息和功能,确保客户端能够访问到必要的子系统功能。

提供清晰的文档

  • 文档说明:为外观类提供清晰的文档,说明其提供的功能和使用方法。

考虑线程安全

  • 并发访问:如果外观类可能会被多个线程访问,确保其方法是线程安全的。

灵活使用

  • 适应变化:设计外观类时,考虑未来可能的变化,使其能够灵活适应子系统的扩展和修改。

6.2 避免滥用

避免创建过多的外观类

  • 合理使用:只在确实需要简化复杂系统接口时使用外观模式,避免创建不必要的外观类。

避免过度集中逻辑

  • 逻辑分散:避免将过多的业务逻辑集中在外观类中,这可能会导致外观类变得复杂和难以维护。

避免过度隐藏细节

  • 适度隐藏:适度隐藏实现细节,确保客户端能够访问到必要的功能和信息。

6.3 替代方案

使用依赖注入

  • 依赖注入:通过依赖注入来管理对象的依赖关系,而不是通过外观类来协调子系统。

使用中介者模式

  • 中介者模式:当系统中的对象之间存在复杂的通信时,可以使用中介者模式来简化它们之间的交互。

使用命令模式

  • 命令模式:当需要对操作进行封装和调度时,可以使用命令模式来实现。

使用策略模式

  • 策略模式:当需要根据不同的策略动态改变对象行为时,可以使用策略模式。

使用组合模式

  • 组合模式:当需要表示部分-整体层次结构时,可以使用组合模式。

外观模式是一种有用的设计模式,可以简化复杂系统的访问并降低系统间的耦合度。然而,合理使用外观模式并避免其缺点是至关重要的。了解其替代方案可以帮助开发者根据具体需求和场景选择最合适的设计模式。在实际开发中,应根据具体情况灵活运用外观模式,以达到最佳的设计效果。

结语

外观模式提供了一种简化复杂系统访问的方法,使得客户端可以更容易地与复杂的子系统交互。通过本文的深入分析,希望读者能够对外观模式有更全面的理解,并在实际开发中做出合理的设计选择。

 相关Java设计模式文章推荐:

Java二十三种设计模式-单例模式(1/23)

Java二十三种设计模式-工厂方法模式(2/23)

Java二十三种设计模式-抽象工厂模式(3/23)

Java二十三种设计模式-建造者模式(4/23)

Java二十三种设计模式-原型模式(5/23)

Java二十三种设计模式-适配器模式(6/23)

Java二十三种设计模式-装饰器模式(7/23)

Java二十三种设计模式-代理模式(8/23)

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

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

相关文章

Android 10.0 Launcher3仿ios的folder文件夹widget功能实现二

1.前言 在10.0的系统ROM开发中,在进行一些系统Launcher3定制功能开发中,需要实现folder文件夹widget的功能,由于launcher3 默认不支持folder跨行显示,所以就需要借助自定义的widget小部件功能来实现相关功能,接下来分析实现相关功能 2.Launcher3仿ios的folder文件夹widge…

jQuery前端网页制作

1、Jquery的概述 1.1JavaScript库 JavaScript 高级程序设计(特别是对浏览器差异的复杂处理),通常很困难也很耗时。 为了应对这些调整,许多的 JavaScript (helper) 库应运而生。 这些 JavaScript 库常被称为 JavaScript 框架。 市面上一些广受欢迎的 JavaScript 框架:…

大厂linux面试题攻略五之数据库管理

一、数据库管理-MySQL语句 0.MySQL基本语句: 1.SQL语句-增 创建xxx用户: mysql>create user xxx % indentified by 123456; xxx表示用户名 %b表示该用户用来连接数据库的方式(远程或本地连接) indentified by 123456设置密码…

Reranker技术

文章目录 Reranker技术0. 什么是RAG1. 什么是Reranker?2. Reranker在RAG技术中的应用3.使用 Reranker 的优缺点4.总结参考:知乎 Reranker技术 0. 什么是RAG 基础 RAG 的操作流程大致如下:首先,你需要将文本切分成小段&#xff0…

centos7 docker空间不足

今天在使用docker安装镜像的时候,出现报错 查看原因,发现是分区空间不足导致的 所以考虑进行扩容 首先在vmware扩容并没有生效 因为只是扩展的虚拟空间,并不支持扩展分区大小,下面对分区进行扩容 参考: 分区扩容 主…

细说MCU的DAC改变输出信号频率的方法

目录 一、参考硬件 二、改变输出信号的频率 1.建立新工程 2.配置TIM3 三、代码修改 四、查看结果 一、参考硬件 本项目依赖的软件和硬件工程参考本文作者写的文章:细说MCU的DAC1和DAC2各自输出一通道模拟信号的方法-CSDN博客 https://wenchm.blog.csdn.net/a…

【初阶数据结构篇】二叉树算法题

文章目录 二叉树算法题前言单值二叉树相同的树对称二叉树另一棵树的子树二叉树的前序遍历 二叉树算法题 前言 本篇的算法题涉及到链式结构二叉树的实现方法可参考:二叉链实现方法上篇二叉链实现方法下篇 单值二叉树 如果二叉树每个节点都具有相同的值,…

什么情况?我代码没了

前两天检视代码时,发现PR里面有两个提交的描述信息一模一样,于是我提出应该将这两个提交合并成一个,保持提交树的清晰。 1 先储存起来! 而同事这时正在开发别的特性,工作区不是干净的,没法直接执行 git r…

APDL(ANSYS Parametric Design Language)初识

APDL(ANSYS Parametric Design Language)编写涉及使用ANSYS的参数化设计语言来创建、修改和执行有限元分析(FEA)任务。以下是一些关于APDL编写的基本步骤、技巧和示例: 一、基本步骤 了解APDL基础: 熟悉AP…

如何在 Kali Linux 上安装和使用 Docker 和 Docker Compose

Docker 和 Docker Compose 是现代开发者必备的工具,特别是当你需要在不同的环境中部署应用时。本文将详细介绍如何在 Kali Linux 上安装 Docker 和 Docker Compose,并使用它们启动服务。即使你是个技术小白,也能轻松跟随这篇指南完成操作。 …

轻松搞定 Nginx 在 CentOS 和 Ubuntu 上的安装与配置

注:这是对我以前博客进行优化后再次发布的,博客中的截图为以前的。原博客已删除。 如何安装nginx nginx是一款开源、高性能的Web和反向代理服务器,支持HTTP、HTTPS、SMTP、POP3和IMAP协议。由于其轻量级、资源占用少和强大的并发能力&#…

基于vue2 + Ant Design 封装input(输入)下拉Table表格

封装 AInputTable 组件 <!--下拉Table--> <template><div class"input-select-table" ref"inputTableRef" v-clickoutside"handleHide"><div class"input-select-table-input" click"disabled?this:hand…

【C++] 认识C++(二)

前言 &#x1f4da;作者简介&#xff1a;爱编程的小马&#xff0c;是一名大厂后端c程序员。 &#x1f4da;本文收录于C系列&#xff0c;本专栏主要是分享我所了解的c知识&#xff0c;带领大家慢慢从了解c到认识c&#xff0c;持续更新&#xff01; &#x1f4da;相关专栏Linux正…

Base64解码时Illegal base64 character 20问题解决

一&#xff0c;问题 在使用Base64解码的时候 // 这里的keyContent是公钥&#xff0c;一般配置到配置中心里&#xff0c;然后注入到容器里 String publicKeyString keyContent .replaceAll("\\n", "") .replace("-----BEGIN PUBLIC KEY-----",…

sqli-labs(6-10)关通关讲解

sqli-labs(6-10)关通关讲解 Less-6 方法一&#xff1a;手工注入 1.判断闭合 http://localhost/sqli-labs/Less-6/?id1" //报错 http://localhost/sqli-labs/Less-6/?id1" -- //正常 http://localhost/sqli-labs/Less-6/?id1" and 11 -- http://localhos…

Vue Amazing UI:高颜值、高性能的前端组件库

Vue Amazing UI&#xff1a;高颜值、高性能的前端组件库 在当今前端开发中&#xff0c;Vue Amazing UI 作为一款功能强大的 UI 组件库&#xff0c;为开发者提供了全面的解决方案。本文将介绍 Vue Amazing UI 的基本信息、特点以及如何快速部署和使用。 软件简介 Vue Amazing U…

Win11没有记事本怎么办?更新至win11无法右键新建txt文件?

博主更新至Win11系统后目前用了不到一个月时间&#xff0c;今天突然发现 鼠标右键无法新建txt文件 了&#xff0c;一开始还以为Win11系统不支持txt类型文件&#xff0c;遂查找各种网上恢复教程。本文综合了多篇教程的方法&#xff0c;力求一文解决所有可能出现的情况&#xff0…