外观模式:简化复杂系统接口的设计模式
外观模式:简化复杂系统接口的设计模式
一、模式核心:为复杂子系统提供统一简单接口
当一个系统由多个复杂子系统组成时(如电商系统中的支付、物流、库存模块),客户端直接调用子系统会导致依赖关系复杂、代码难以维护。
外观模式(Facade Pattern) 通过引入一个外观类(Facade),将子系统的复杂接口封装为一个简单的统一接口,客户端只需与外观类交互,无需了解子系统的内部细节。核心解决:
- 解耦客户端与子系统:降低客户端对多个子系统的直接依赖,简化调用流程。
- 隐藏系统复杂性:将子系统的交互细节封装在外观类中,提供清晰易用的高层接口。
- 分层设计支持:在架构中划分层次(如门面层、服务层),符合迪米特法则(最少知识原则)。
核心思想与 UML 类图
外观模式包含以下角色:
- 外观类(Facade):定义客户端调用的简单接口,内部调用子系统的功能。
- 子系统类(Subsystem Classes):实现系统的具体功能,相互之间可能有复杂交互。
- 客户端(Client):通过外观类间接使用子系统功能。
二、核心实现:电商下单流程的外观封装
1. 定义子系统(支付、物流、库存)
支付模块
public class PaymentSystem { public void processPayment(double amount) { System.out.println("支付模块:处理支付,金额 " + amount + " 元"); }
}
物流模块
public class LogisticsSystem { public void shipOrder(String address) { System.out.println("物流模块:发货至地址 " + address); }
}
库存模块
public class InventorySystem { public boolean checkStock(int productId) { System.out.println("库存模块:检查商品 " + productId + " 库存"); return true; // 假设库存充足 } public void deductStock(int productId, int quantity) { System.out.println("库存模块:扣除商品 " + productId + " 库存 " + quantity + " 件"); }
}
2. 实现外观类(电商门面)
public class ECommerceFacade { private PaymentSystem paymentSystem = new PaymentSystem(); private LogisticsSystem logisticsSystem = new LogisticsSystem(); private InventorySystem inventorySystem = new InventorySystem(); // 统一下单接口,封装三个子系统调用 public void placeOrder(int productId, int quantity, double price, String address) { // 1. 检查库存 if (!inventorySystem.checkStock(productId)) { System.out.println("下单失败:库存不足"); return; } // 2. 扣除库存 inventorySystem.deductStock(productId, quantity); // 3. 处理支付 double totalAmount = price * quantity; paymentSystem.processPayment(totalAmount); // 4. 安排发货 logisticsSystem.shipOrder(address); System.out.println("下单成功!"); }
}
3. 客户端通过外观类下单
public class ClientDemo { public static void main(String[] args) { ECommerceFacade facade = new ECommerceFacade(); // 只需调用外观类的统一接口,无需了解子系统细节 facade.placeOrder( productId = 1001, quantity = 2, price = 99.9, address = "北京市朝阳区 XX 路" ); }
}
输出结果:
库存模块:检查商品 1001 库存
库存模块:扣除商品 1001 库存 2 件
支付模块:处理支付,金额 199.8 元
物流模块:发货至地址 北京市朝阳区 XX 路
下单成功!
三、进阶:外观模式与适配器模式结合
当子系统接口与外观接口不兼容时,可通过适配器模式转换接口,使外观类能适配不同子系统。
1. 定义旧版支付接口(需适配)
public class LegacyPaymentSystem { public void pay(double amount, String method) { System.out.println("旧版支付:使用 " + method + " 支付 " + amount + " 元"); }
}
2. 实现适配器类
public class PaymentAdapter extends PaymentSystem { private LegacyPaymentSystem legacySystem = new LegacyPaymentSystem(); @Override public void processPayment(double amount) { legacySystem.pay(amount, "支付宝"); // 适配旧版接口 }
}
3. 修改外观类以支持适配器
public class ECommerceFacade { private PaymentSystem paymentSystem = new PaymentAdapter(); // 使用适配器 // ... 其他代码不变 ...
}
四、框架与源码中的外观实践
1. Spring MVC 的 DispatcherServlet
Spring MVC 通过 DispatcherServlet
作为外观类,统一处理客户端请求,调用控制器、模型、视图等子模块,隐藏请求分发、参数解析等细节。
2. AWT/Swing 的图形界面封装
Java 图形库通过外观类(如 JFrame
、JPanel
)封装底层操作系统的图形接口,实现跨平台的统一界面展示。
3. MyBatis 的 SqlSession
MyBatis 的 SqlSession
作为外观接口,封装了连接获取、SQL 执行、结果映射等子系统操作,客户端只需调用简单方法(如 selectOne()
)即可完成数据库操作。
五、避坑指南:正确使用外观模式的 3 个要点
1. 避免外观类成为万能接口
外观类应聚焦于封装 “高频组合操作”,而非包含所有子系统功能。若外观类过于臃肿,可拆分为多个专用外观类(如订单外观、支付外观)。
2. 保留对子系统的直接访问
外观模式不强制客户端只能通过外观类访问子系统。若客户端需要细粒度控制,仍可直接调用子系统接口(如复杂业务场景下绕过外观类)。
3. 区分外观模式与中介者模式
- 外观模式:简化接口,不涉及子系统间的交互协调(子系统间可直接调用)。
- 中介者模式:解耦子系统,所有交互通过中介者完成(子系统间不直接通信)。
六、总结:何时该用外观模式?
适用场景 | 核心特征 | 典型案例 |
---|---|---|
复杂系统对外接口 | 系统由多个子系统组成,需提供简单入口 | 微服务网关、系统集成平台 |
旧系统迁移 | 封装旧系统接口,适配新系统需求 | 遗留系统对接、第三方服务集成 |
分层架构设计 | 划分门面层与业务层,降低层间耦合 | MVC 模式中的控制器层、DDD 中的应用层 |
外观模式通过 “统一接口 + 隐藏细节” 的设计,显著提升了系统的易用性和可维护性。下一篇我们将探讨备忘录模式,解析如何实现对象状态的撤销与恢复,敬请期待!
扩展思考:外观模式的优点
- 提高开发效率:客户端无需了解子系统细节,只需调用外观接口,缩短开发周期。
- 增强可测试性:通过外观类模拟子系统行为,便于单元测试(如使用 Mock 对象替代真实子系统)。
相关文章:

外观模式:简化复杂系统接口的设计模式
外观模式:简化复杂系统接口的设计模式 一、模式核心:为复杂子系统提供统一简单接口 当一个系统由多个复杂子系统组成时(如电商系统中的支付、物流、库存模块),客户端直接调用子系统会导致依赖关系复杂、代码难以维护…...
vue3项目中eslint.config.ts配置rules
vue3项目中eslint.config.ts配置rules 1. 使用npm create vuelatest创建vue项目 默认的eslint.config.ts如下 import { globalIgnores } from eslint/config import { defineConfigWithVueTs, vueTsConfigs } from vue/eslint-config-typescript import pluginVue from esli…...

uniapp-商城-36-shop 购物车 选好了 进行订单确认2 支付方式颜色变化和颜色滤镜filter
颜色滤镜,在好多网页都这样使用,滤掉彩色,显示黑白,这在一些关键的日子中都这样使用。 1、依然回到订单确认页面 看到支付的颜色了嘛? <view class"payType"><view class"box" :class&q…...

Vue3 上传后的文件智能预览(实战体会)
目录 前言1. Demo12. Demo2 前言 🤟 找工作,来万码优才:👉 #小程序://万码优才/r6rqmzDaXpYkJZF 爬虫神器,无代码爬取,就来:bright.cn 此处的基本知识涉及较少,主要以Demo的形式供大…...
铃木一郎女儿是奥运会选手吗·棒球1号位
铃木一朗(Ichiro Suzuki) 铃木一朗职业生涯时间线 1973年出生于日本爱知县名古屋市。1992年以选秀第四顺位加入日本职棒(NPB)欧力士蓝浪队,开启职业棒球生涯。 1994-2000年 连续7年获得NPB太平洋联盟打击王ÿ…...
PyTorch与CUDA的关系
文章目录 前言一、如何查看PyTorch和torchvision的版本1.1 查看PyTorch版本1.2 查看torchvision版本二、如何确认PyTorch和torchvision是否支持CUDA加速2.1 检查PyTorch是否支持CUDA2.2 查看当前可用的GPU设备2.3 检查torchvision是否支持CUDA三、CUDA版本的秘密:为什么PyTorc…...

CCE13.【C++ Cont】练习题组13 静态链表专题
目录 1.B3630 排队顺序 题目 分析 代码 提交结果 2.B3631 单向链表 题目 分析 前置知识:map数组加快访问速度(简单的哈希表优化) 使用map数组的重要提醒 代码 提交结果 3.★P1160 队列安排 题目 分析 方法1:带头不循环双向链表的设计 方法2:带头循环的双向链表…...
【Mybatis】MyBatisPlus的saveBatch真的是批量插入吗?深度解析与性能优化
前言 在使用MyBatis-Plus进行批量数据插入时,许多开发者会发现:即使调用saveBatch方法,数据库仍会产生大量INSERT语句。本文将深入源码揭示背后的真相,并提供3种性能优化方案,让你的批量插入速度提升10倍!…...

内联函数(c++)
预处理:优点:内嵌到目标代码,减少函数的调用。 缺点:在预处理阶段完成替换,避免了语义上的差错。 egg: #define SQR(X) ((X)*(X)) 函数:优点:完成了某一类操作的抽象,…...

R7周:糖尿病预测模型优化探索
🍨 本文为🔗365天深度学习训练营中的学习记录博客 🍖 原作者:K同学啊 一、数据预处理 1.设置GPU import torch.nn.functional as F import torch.nn as nn import torch, torchvisiondevice torch.device("cuda"…...

线程怎么创建?Java 四种方式一网打尽
🚀 Java 中线程的 4 种创建方式详解 创建方式实现方式是否推荐场景说明1. 继承 Thread 类class MyThread extends Thread❌ 不推荐简单学习、单线程场景2. 实现 Runnable 接口class MyRunnable implements Runnable✅ 推荐更适合多线程共享资源3. 实现 Callable 接…...
前端如何连接tcp 服务,接收数据
在传统的浏览器前端环境中,由于浏览器的同源策略和安全限制,无法直接建立 TCP 连接。不过,可以通过 WebSocket 或者使用 WebRTC 来间接实现与 TCP 服务的通信,另外在 Node.js 环境中可以直接使用 net 模块建立 TCP 连接。下面分别…...

STM32之DHT11温湿度传感器---附代码
DHT11简介 DHT11的供电电压为 3-5.5V。 传感器上电后,要等待 1s 以越过不稳定状态在此期间无需发送任何指令。 电源引脚(VDD,GND)之间可增加一个100nF 的电容,用以去耦滤波。 DATA 用于微处理器与DHT11之间…...

工业相机——镜头篇【机器视觉,图像采集系统,成像原理,光学系统,成像光路,镜头光圈,镜头景深,远心镜头,分辨率,MTF曲线,焦距计算 ,子午弧矢】
文章目录 1 机器视觉,图像采集系统2 相机镜头,属于一种光学系统3 常规镜头 成像光路4 镜头光圈5 镜头的景深6 远心镜头 及 成像原理7 远心镜头种类 及 应用场景8 镜头分辨率10 镜头的对比度11 镜头的MTF曲线12 镜头的焦距 计算13 子午弧矢 图解 反差 工业…...
如何在Spring Boot中禁用Actuator端点安全性
在 Spring Boot 应用中,Spring Boot Actuator 提供了一系列用于监控和管理应用的端点(如 /actuator/health、/actuator/metrics),这些端点默认可能受到 Spring Security 的保护,要求身份验证或授权。然而,在…...
第48讲:空间大数据与智慧农业——时空大数据分析与农业物联网的融合实践
目录 🧠 一、什么是空间大数据? 📡 二、农业物联网:数据采集的神经末梢 🔁 三、融合应用:空间大数据 + 农业IoT = 决策大脑 1. 精准灌溉管理 2. 时空病虫害预警 3. 农业碳监测与生态评估 💡 四、技术实践案例:农田干旱预警系统 📌 场景设定: 🛠 数据…...

openwrt查询网关的命令
方法一:route -n 方法二:ip route show...

华为OD机试真题——查找接口成功率最优时间段(2025A卷:100分)Java/python/JavaScript/C/C++/GO最佳实现
2025 A卷 100分 题型 本专栏内全部题目均提供Java、python、JavaScript、C、C、GO六种语言的最佳实现方式; 并且每种语言均涵盖详细的问题分析、解题思路、代码实现、代码详解、3个测试用例以及综合分析; 本文收录于专栏:《2025华为OD真题目录…...
SiamMask原理详解:从SiamFC到SiamRPN++,再到多任务分支设计
SiamMask原理详解:从SiamFC到SiamRPN,再到多任务分支设计 一、引言二、SiamFC:目标跟踪的奠基者1. SiamFC的结构2. SiamFC的局限性 三、SiamRPN:引入Anchor机制的改进1. SiamRPN的创新2. SiamRPN的进一步优化 四、SiamMask&#x…...
Gradle安装与配置国内镜像源指南
一、Gradle简介与安装准备 Gradle是一款基于JVM的现代化构建工具,广泛应用于Java、Kotlin、Android等项目的构建自动化。相比传统的Maven和Ant,Gradle采用Groovy或Kotlin DSL作为构建脚本语言,具有配置灵活、性能优越等特点。 在开始安装前…...

【“星睿O6”AI PC开发套件评测】开箱+刷机+基础环境配置
开箱 很荣幸可以参与“星睿O6”AI PC开发套件评测,话不多说先看开箱美图,板子的包装还是蛮惊艳的。 基础开发环境配置 刷机 刷机参考这里的文档快速上手即可,笔者同时验证过使用USB和使用NVMe硬盘盒直接在硬盘上刷机,操作下来建…...

力扣面试150题--环形链表和两数相加
Day 32 题目描述 思路 采取快慢指针 /*** Definition for singly-linked list.* class ListNode {* int val;* ListNode next;* ListNode(int x) {* val x;* next null;* }* }*/ public class Solution {public boolean hasCycle(ListNod…...
Dapper的数据库操作备忘
Dapper是很好的C#生态的ORM工具 获取单条记录 var row conn.QueryFirstOrDefault("select abc as cc"); if (row null) return; string priField row.cc; //直接访问字段根据动态的字段名获取值,则需要先转为字典接口 var dict (IDictionary<string, objec…...

STM32 TIM输入捕获
一、输入捕获简介 IC(Input Capture)输入捕获输入捕获模式下,当通道输入引脚出现指定电平跳变时,当前CNT的值将被锁存到CCR中,可用于测量PWM波形的频率、占空比、脉冲间隔、电平持续时间等参数每个高级定时器和通用定…...

python项目实战-后端个人博客系统
本文分享一个基于 Flask 框架开发的个人博客系统后端项目,涵盖用户注册登录、文章发布、分类管理、评论功能等核心模块。适合初学者学习和中小型博客系统开发。 一、项目结构 blog │ app.py │ forms.py │ models.py │ ├───instance │ blog.d…...

白鲸开源与亚马逊云科技携手推动AI-Ready数据架构创新
在昨日举办的2025亚马逊云科技合作伙伴峰会圆桌论坛上,白鲸开源创始人兼CEO郭炜作为嘉宾,与亚马逊云科技及其他行业领袖共同探讨了“AI-Ready的数据架构:ISV如何构建面向生成式AI的强大数据基座”这一重要话题。此次论坛由亚马逊云科技大中华…...
【目标检测】目标检测综述 目标检测技巧
I. 目标检测中标注的关键作用 A. 目标检测数据标注的定义 目标检测是计算机视觉领域的一项基础且核心的任务,其目标是在图像或视频中准确识别并定位出预定义类别的目标实例 1。数据标注,在目标检测的语境下,指的是为原始视觉数据࿰…...
[AI技术(二)]JSONRPC协议MCPRAGAgent
Agent概述(一) AI技术基础(一) JSON-RPC 2.0 协议详解 JSON-RPC 2.0 是一种基于 JSON 的轻量级远程过程调用(RPC)协议,旨在简化跨语言、跨平台的远程通信。以下从协议特性、核心结构、错误处理、批量请求等角度进行详细解析: 一、协议概述 1. 设计原则 • 简单性:…...

探秘LLM推理模型:hidden states中藏着的self verification的“钥匙”
推理模型在数学和逻辑推理等任务中表现出色,但常出现过度推理的情况。本文研究发现,推理模型的隐藏状态编码了答案正确性信息,利用这一信息可提升推理效率。想知道具体如何实现吗?快来一起来了解吧! 论文标题 Reasoni…...

大数据开发环境的安装,配置(Hadoop)
1. 三台linux服务器的安装 1. 安装VMware VMware虚拟机软件是一个“虚拟PC”软件,它使你可以在一台机器上同时运行二个或更多Windows、DOS、LINUX系统。与“多启动”系统相比,VMWare采用了完全不同的概念。 我们可以通过VMware来安装我们的linux虚拟机…...