JavaScript设计模式 -- 外观模式

在实际开发中,往往会遇到多个子系统协同工作时,直接操作各个子系统不仅接口繁琐,还容易导致客户端与内部实现紧密耦合。**外观模式(Facade Pattern)**通过为多个子系统提供一个统一的高层接口,将复杂性隐藏在内部,从而降低耦合,提高代码的可维护性与易用性。

本文将详细介绍外观模式的基本概念、结构和优缺点,并通过多个示例展示如何在不同场景下运用外观模式。

外观模式简介

外观模式属于结构型设计模式,其核心思想是为多个复杂子系统提供一个统一的接口(外观),从而让客户端无需了解内部实现细节即可调用系统功能。外观模式不仅能简化调用流程,还能使系统的内部变化对客户端透明,便于后续扩展和维护。


外观模式的结构与特点

主要角色:

  • 外观(Facade)
    对外提供一个统一的接口,封装多个子系统的功能调用。

  • 子系统(Subsystem)
    完成具体的业务逻辑,各自拥有独立的接口和实现。

  • 客户端(Client)
    只需通过外观接口与系统交互,无需关心子系统内部的复杂细节。

特点:

  • 简化接口:将复杂的操作组合成简单的方法调用。
  • 降低耦合:客户端只依赖外观接口,而不直接与各个子系统耦合。
  • 隐藏复杂性:将系统内部的实现细节封装在外观类中,对外部屏蔽。

多方面示例详解

下面通过多个示例,展示如何利用外观模式解决不同场景下的复杂性问题。

示例 1:家庭影院系统

在家庭影院中,通常需要协调音响、投影仪、灯光、蓝光播放器等多个设备。直接控制这些设备非常繁琐,使用外观模式可以为家庭影院提供一键启动和关闭的简单接口。

// 子系统:音响
class Amplifier {on() {console.log('音响开启');}off() {console.log('音响关闭');}setVolume(level) {console.log(`音响音量设置为 ${level}`);}
}// 子系统:投影仪
class Projector {on() {console.log('投影仪开启');}off() {console.log('投影仪关闭');}setInput(input) {console.log(`投影仪输入源设置为 ${input}`);}
}// 子系统:灯光
class TheaterLights {dim(level) {console.log(`灯光调暗到 ${level}%`);}on() {console.log('灯光开启');}
}// 子系统:蓝光播放器
class BluRayPlayer {on() {console.log('蓝光播放器开启');}off() {console.log('蓝光播放器关闭');}play(movie) {console.log(`正在播放电影:《${movie}》`);}
}// 外观类:家庭影院外观
class HomeTheaterFacade {constructor(amp, projector, lights, bluRay) {this.amp = amp;this.projector = projector;this.lights = lights;this.bluRay = bluRay;}watchMovie(movie) {console.log('准备观看电影...');this.lights.dim(10);this.projector.on();this.projector.setInput('蓝光播放器');this.amp.on();this.amp.setVolume(5);this.bluRay.on();this.bluRay.play(movie);}endMovie() {console.log('结束观看电影...');this.lights.on();this.projector.off();this.amp.off();this.bluRay.off();}
}// 客户端调用
const amplifier = new Amplifier();
const projector = new Projector();
const lights = new TheaterLights();
const bluRayPlayer = new BluRayPlayer();const homeTheater = new HomeTheaterFacade(amplifier, projector, lights, bluRayPlayer);
homeTheater.watchMovie('阿凡达');
homeTheater.endMovie();

示例 2:电子商务系统统一接口

在电子商务平台中,下单流程可能涉及订单创建、支付处理、物流安排和通知发送等多个子系统。使用外观模式,可以将这些流程封装为一个简单的 placeOrder 接口。

// 子系统:订单处理
class OrderService {createOrder(orderDetails) {console.log(`订单已创建:${JSON.stringify(orderDetails)}`);return 'ORDER123';}
}// 子系统:支付系统
class PaymentService {processPayment(orderId, amount) {console.log(`订单 ${orderId} 支付金额 ${amount} 元成功`);}
}// 子系统:物流系统
class ShippingService {arrangeShipping(orderId) {console.log(`订单 ${orderId} 发货成功`);}
}// 子系统:通知系统
class NotificationService {sendNotification(message) {console.log(`发送通知:${message}`);}
}// 外观类:购物流程外观
class ShoppingFacade {constructor(orderService, paymentService, shippingService, notificationService) {this.orderService = orderService;this.paymentService = paymentService;this.shippingService = shippingService;this.notificationService = notificationService;}placeOrder(orderDetails, amount) {console.log('开始下单流程...');const orderId = this.orderService.createOrder(orderDetails);this.paymentService.processPayment(orderId, amount);this.shippingService.arrangeShipping(orderId);this.notificationService.sendNotification(`订单 ${orderId} 已完成处理`);}
}// 客户端调用
const orderService = new OrderService();
const paymentService = new PaymentService();
const shippingService = new ShippingService();
const notificationService = new NotificationService();const shopping = new ShoppingFacade(orderService, paymentService, shippingService, notificationService);
shopping.placeOrder({ item: '笔记本电脑', quantity: 1 }, 8000);

示例 3:网络请求聚合接口

在一个大型 Web 应用中,客户端可能需要从多个 API 接口获取数据(例如用户信息、订单信息、统计数据等)。通过外观模式,可以设计一个统一的 API 聚合层,对外暴露简洁的接口,而内部调用各个子 API。

// 子系统:用户服务
class UserService {fetchUser(userId) {console.log(`获取用户 ${userId} 信息...`);return { id: userId, name: '张三' };}
}// 子系统:订单服务
class OrderServiceAPI {fetchOrders(userId) {console.log(`获取用户 ${userId} 的订单...`);return [{ orderId: 'ORDER123', item: '手机' }];}
}// 子系统:统计服务
class StatsService {fetchStats(userId) {console.log(`获取用户 ${userId} 的统计数据...`);return { totalOrders: 5, totalSpent: 1200 };}
}// 外观类:API 聚合器
class APIServiceFacade {constructor(userService, orderService, statsService) {this.userService = userService;this.orderService = orderService;this.statsService = statsService;}getUserDashboard(userId) {const user = this.userService.fetchUser(userId);const orders = this.orderService.fetchOrders(userId);const stats = this.statsService.fetchStats(userId);return {user,orders,stats};}
}// 客户端调用
const userServiceInstance = new UserService();
const orderServiceInstance = new OrderServiceAPI();
const statsServiceInstance = new StatsService();const apiFacade = new APIServiceFacade(userServiceInstance, orderServiceInstance, statsServiceInstance);
const dashboard = apiFacade.getUserDashboard(101);
console.log('用户仪表盘数据:', dashboard);

示例 4:游戏初始化外观

在游戏开发中,启动一个游戏往往需要初始化多个子系统,如图形引擎、音频系统、输入管理等。通过外观模式,可以为游戏引擎提供一个统一的初始化接口,简化启动流程。

// 子系统:图形引擎
class GraphicsEngine {init() {console.log('图形引擎初始化完成');}
}// 子系统:音频系统
class AudioSystem {init() {console.log('音频系统初始化完成');}
}// 子系统:输入管理
class InputManager {init() {console.log('输入管理初始化完成');}
}// 外观类:游戏引擎外观
class GameEngineFacade {constructor(graphics, audio, input) {this.graphics = graphics;this.audio = audio;this.input = input;}initializeGame() {console.log('游戏启动中...');this.graphics.init();this.audio.init();this.input.init();console.log('游戏初始化完毕');}
}// 客户端调用
const graphicsEngine = new GraphicsEngine();
const audioSystem = new AudioSystem();
const inputManager = new InputManager();const gameEngine = new GameEngineFacade(graphicsEngine, audioSystem, inputManager);
gameEngine.initializeGame();

示例 5:跨平台资源加载

在移动开发或跨平台项目中,不同平台可能需要加载不同的资源或配置。利用外观模式可以创建一个资源加载外观,根据当前平台选择合适的加载器,从而对外提供统一的加载接口。

// 子系统:Android 资源加载器
class AndroidResourceLoader {loadResources() {console.log('加载 Android 平台资源...');}
}// 子系统:iOS 资源加载器
class IOSResourceLoader {loadResources() {console.log('加载 iOS 平台资源...');}
}// 外观类:跨平台资源加载器
class ResourceFacade {constructor(platform) {// 假设 platform 值为 'android' 或 'ios'this.loader = platform === 'android'? new AndroidResourceLoader(): new IOSResourceLoader();}load() {this.loader.loadResources();}
}// 客户端调用
const currentPlatform = 'ios'; // 模拟当前平台为 iOS
const resourceLoader = new ResourceFacade(currentPlatform);
resourceLoader.load();

外观模式的优缺点

优点

  • 简化接口:将多个子系统的调用封装成一个简单接口,降低使用复杂度。
  • 降低耦合:客户端与各子系统解耦,任何子系统的变化只需在外观层做适配。
  • 隐藏内部复杂性:外观模式屏蔽了子系统实现细节,使得系统更易于使用和维护。

缺点

  • 灵活性降低:外观模式封装了子系统的所有功能,可能限制了对某些细节的直接控制。
  • 外观类可能过于庞大:当涉及的子系统很多时,外观类的职责可能变得过于繁杂,需要合理设计职责分离。

总结

外观模式通过为复杂系统提供一个统一而简洁的接口,有效降低了客户端与各子系统之间的耦合,使得系统调用更加直观和易于维护。本文通过家庭影院、电子商务、网络请求聚合、游戏初始化和跨平台资源加载五个示例,展示了外观模式在不同场景下的应用。希望这些实例能够帮助你在实际项目中发现并利用外观模式带来的简化接口和隐藏复杂性的优势。

欢迎在评论区分享你的使用心得或疑问!

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Chrome多开终极形态解锁!「窗口管理工具+IP隔离插件

Web3项目多开,继ads指纹浏览器钱包被盗后,更多人采用原生chrome浏览器,当然对于新手,指纹浏览器每月成本也是一笔不小开支,今天逛Github发现了这样一个解决方案,作者开发了窗口管理工具IP隔离插件&#xff…

从零开始部署DeepSeek:基于Ollama+Flask的本地化AI对话系统

从零开始部署DeepSeek:基于OllamaFlask的本地化AI对话系统 一、部署背景与工具选型 在AI大模型遍地开花的2025年,DeepSeek R1凭借其出色的推理能力和开源特性成为开发者首选。本文将以零基础视角,通过以下工具链实现本地化部署: …

python旅游推荐系统+爬虫+可视化(协同过滤算法)

✅️基于用户的协同过滤算法 ✅️有后台管理 ✅️2w多数据集 这个旅游数据分析推荐系统采用了Python语言、Django框架、MySQL数据库、requests库进行网络爬虫开发、机器学习中的协同过滤算法、ECharts数据可视化技术,以实现从网站抓取旅游数据、个性化推荐和直观展…

以 Serverless 低成本的⽅式 快速在亚马逊云科技上部署 DeepSeek

2025年春节,最令人瞩目的无疑是DeepSeek的惊艳亮相,它以颠覆性的创新迅速席卷全球,成为街谈巷议的热点。无论是在地铁车厢里,还是公司茶水间,DeepSeek都成了人们津津乐道的话题。社交平台上,网友们争相分享…

win10 系统 自定义Ollama安装路径 及模型下载位置

win10 系统 自定义Ollama安装路径 及模型下载位置 由于Ollama的exe安装软件双击安装的时候默认是在C盘,以及后续的模型数据下载也在C盘,导致会占用C盘空间,所以这里单独写了一个自定义安装Ollama安装目录的教程。 Ollama官网地址&#xff1…

CAP与BASE:分布式系统设计的灵魂与妥协

CAP 理论 CAP理论起源于 2000 年,由加州大学伯克利分校的 Eric Brewer 教授在分布式计算原理研讨会(PODC)上提出,因此 CAP 定理又被称作 布鲁尔定理(Brewer’s theorem) 2 年后,麻省理工学院的 …

电动汽车电池监测平台系统设计(论文+源码+图纸)

1总体设计 本次基于单片机的电池监测平台系统设计,其整个系统架构如图2.1所示,其采用STC89C52单片机作为控制器,结合ACS712电流传感器、TLC1543模数转换器、LCD液晶、DS18B20温度传感器构成整个系统,在功能上可以实现电压、电流、…

docker下部署kong+consul+konga 报错问题处理

前言: 由于在docker下部署一些项目比较特殊,特别是网络这一块,如果没有搞清楚,是很容易出问题的。 先上docker-compose 编排 这里的docker-compose for kong可以在 kong-compose 获取代码 version: 3.9x-kong-config:&kong…

装饰器模式

参考 装饰者模式 【设计模式实战】装饰器模式 1. HistorySet的例子 HistorySet 可以在实现的Set的基础上&#xff0c;在remove时保留删除的元素。通过将方法委托给现有的Set&#xff0c;在remove时先保留被删除元素后委托给注入的set进行remove public class HistorySet<…

软件定义汽车时代的功能安全和信息安全

我是穿拖鞋的汉子&#xff0c;魔都中坚持长期主义的汽车电子工程师。 老规矩&#xff0c;分享一段喜欢的文字&#xff0c;避免自己成为高知识低文化的工程师&#xff1a; 简单&#xff0c;单纯&#xff0c;喜欢独处&#xff0c;独来独往&#xff0c;不易合同频过着接地气的生活…

【Golang】GC探秘/写屏障是什么?

之前写了 一篇【Golang】内存管理 &#xff0c;有了很多的阅读量&#xff0c;那么我就接着分享一下Golang的GC相关的学习。 由于Golang的GC机制一直在持续迭代&#xff0c;本文叙述的主要是Go1.9版本及以后的GC机制&#xff0c;该版本中Golang引入了 混合写屏障大幅度地优化了S…

docker 运行 芋道微服务

jar包打包命令 mvn clean install package -Dmaven.test.skiptrue创建文件夹 docker-ai 文件夹下放入需要jar包的文件夹及 docker-compose.yml 文件 docker-compose.yml 内容&#xff1a;我这里的是ai服务&#xff0c;所以将原先的文件内容做了变更&#xff0c;你们需要用到什…

【苍穹外卖】学习

软件开发整体介绍 作为一名软件开发工程师,我们需要了解在软件开发过程中的开发流程&#xff0c; 以及软件开发过程中涉及到的岗位角色&#xff0c;角色的分工、职责&#xff0c; 并了解软件开发中涉及到的三种软件环境。那么这一小节&#xff0c;我们将从 软件开发流程、角色…

网工项目理论1.7 设备选型

本专栏持续更新&#xff0c;整一个专栏为一个大型复杂网络工程项目。阅读本文章之前务必先看《本专栏必读》。 一.交换机选型要点 制式:盒式交换机/框式交换机。功能:二层交换机/三层交换机。端口密度:每交换机可以提供的端口数量。端口速率:百兆/千兆/万兆。交换容量:交换矩阵…

前端面试技巧与实践

在当今快速发展的互联网行业中&#xff0c;前端开发已经成为了一个至关重要的角色。随着技术的不断进步和用户需求的日益复杂&#xff0c;前端工程师的职责不再仅仅是实现页面的布局和交互&#xff0c;而是需要具备全方位的技术能力和工程思维。根据2023年Stack Overflow的开发…

项目2 数据可视化--- 第十五章 生成数据

数据分析是使用代码来探索数据内的规律和关联。 数据可视化是通过可视化表示来 探索和呈现数据集内的规律。 好的数据可视化&#xff0c;可以发现数据集中未知的规律和意义。 一个流行的工具是Matplotlib&#xff0c;他是一个数据绘图库&#xff1b; 还有Plotly包&#xff…

前端常见面试题-2025

vue4.0 Vue.js 4.0 是在 2021 年 9 月发布。Vue.js 4.0 是 Vue.js 的一个重要版本&#xff0c;引入了许多新特性和改进&#xff0c;旨在提升开发者的体验和性能。以下是一些关键的更新和新特性&#xff1a; Composition API 重构&#xff1a;Vue 3 引入了 Composition API 作为…

python学opencv|读取图像(六十八)使用cv2.Canny()函数实现图像边缘检测

【1】引言 前序学习进程中&#xff0c;在对图像进行边缘识别的基础上&#xff0c;先后进行了边缘轮廓绘制&#xff0c;矩形标注、圆形标注和凸包标注。相关文章包括且不限于&#xff1a; python学opencv|读取图像&#xff08;六十四&#xff09;使用cv2.findContours()函数cv…

C语言基础16:二维数组、字符数组

二维数组 定义 二维数组本质上是一个行列式的组合&#xff0c;也就是说二维数组由行和列两部分组成。属于多维数组&#xff0c;二维数组数据是通过行列进行解读。 二维数组可被视为一个特殊的一维数组&#xff0c;相当于二维数组又是一个一维数组&#xff0c;只不过它的元素…

小爱音箱连接电脑外放之后,浏览器网页视频暂停播放后,音箱整体没声音问题解决

背景 22年买的小爱音箱增强版play&#xff0c;小爱音箱连接电脑外放之后&#xff0c;浏览器网页视频暂停播放后&#xff0c;音箱整体没声音&#xff08;一边打着游戏&#xff0c;一边听歌&#xff0c;一边放视频&#xff0c;视频一暂停&#xff0c;什么声音都没了&#xff0c;…