设计模式10、外观模式Facade

解释说明:外观模式(Facade Pattern)又称为门面模式,属于结构型模式
Façade 为子系统中的一组接口提供了一个统一的高层接口,该接口使得子系统更加容易使用
外观(Facade)角色:为多个子系统对外提供一个共同的接口
子系统(Sub System)角色:实现系统的部分功能,客户可以通过外观角色访问它
优点:
对 Client 屏蔽子系统组件,减少了 Client 处理的对象数目,并使得子系统使用起来更加容易。通过引入外观模式,Client 的代码将变得很简单,与之关联的对象也很少。
实现了子系统与 Client 之间的松耦合关系,这使得子系统的组件变化不会影响到调用它的 Client,只需要调整 Facade 即可。
降低了大型软件系统中的编译依赖性,并简化了系统在不同平台之间的移植过程,因为编译一个子系统一般不需要编译所有其他的子系统。一个子系统的修改对其他子系统没有任何影响,而且子系统内部变化也不会影响到外观对象。
只是提供了一个访问子系统的统一入口,并不影响用户直接使用子系统类。
缺点:
不能很好地限制 Client 使用子系统类,如果对 Client 访问子系统类做太多的限制,则会减少可变性和灵活性。
在不引入抽象外观类的情况下,增加新的子系统可能需要修改 Facade 或 Client 的源代码,违背了“开闭原则”。
适用场景
当要为一个复杂子系统提供一个简单接口时。该接口可以满足大多数用户的需求,而且用户也可以越过外观类直接访问子系统。
Client 与多个子系统之间存在很大的依赖性。引入外观类将子系统与 Client 以及其他子系统解耦,可以提高子系统的独立性和可移植性。
在层次化结构中,可以使用外观模式定义系统中每一层的入口。层与层之间不直接产生联系,而通过外观类建立联系,降低层之间的耦合度。
#pragma once
#include <iostream>
#include <string>
#include <windows.h>
const std::string c_stateToStrCourier[] = { "收到", "验证可达性", "分配人员", "派送包裹", "获取交货确认", "完成" };
const std::string c_stateToStrVendor[] = { "收到", "确认库存", "从仓库得到物品", "包装", "联系快递员", "完成" };
const std::string c_stateToStrOrderTeam[] = { "收到", "确认付款", "联系供应商", "完成" };
const int c_nMsec = 300;  // 休眠时间(毫秒) - Sleep(c_nMsec) 处可以替换为一些有用的代码
// 订单团队
class OrderTeam
{
public:void submitRequest() {m_nState = 0;}// 检测状态bool checkStatus() {std::cout << "订单团队 - 当前状态:" <<  c_stateToStrOrderTeam[m_nState] << std::endl;Sleep(c_nMsec);m_nState++;return (m_nState == Complete);}
private:enum States {Received,  // 收到VerifyPayment,  // 确认付款ContactVendor,  // 联系供应商Complete  // 完成};int m_nState;
};
// 供应商
class Vendor
{
public:void submitRequest() {m_nState = 0;}// 检测状态bool checkStatus() {std::cout << "供应商 - 当前状态:" << c_stateToStrVendor[m_nState] <<  std::endl;Sleep(c_nMsec);m_nState++;return (m_nState == Complete);}
private:enum States {Received,  // 收到VerifyInventory,  // 确认库存GetItemFromWareHouse,  // 从仓库得到物品PackItem,  // 包装ContactCourier,  // 联系快递员Complete  // 完成};int m_nState;
};
// 快递员
class Courier
{
public:// 将请求转发给快递员void submitRequest() {m_nState = 0;}// 检测状态bool checkStatus() {std::cout << "快递员 - 当前状态:" << c_stateToStrCourier[m_nState]  << std::endl;Sleep(c_nMsec);m_nState++;return (m_nState == Complete);}
private:enum States {Received,  // 收到VerifyReachbility,  // 验证可达性AssignPerson,  // 分配人员DispatchPackage,  // 派送包裹GetDeliveryConfirmation,  // 获取交货确认Complete  // 完成};int m_nState;
};#pragma once
#include "sub_system.h"
// 网购外观
class OnlineShoppingFacade
{
public:OnlineShoppingFacade() {m_nCount = 0;}// 返回跟踪次数int followupNum() {return m_nCount;}// 提交订单void submitRequest() {m_nState = 0;}// 跟踪订单bool checkStatus() {// 收到订单请求switch (m_nState) {case Received:m_nState++;// 将请求转发给订单团队m_order.submitRequest();std::cout << "********** 提交给订单团队,跟踪次数:" <<  m_nCount << " **********" << std::endl;break;case SubmittedToOrderTeam:// 如果订单团队完成验证,则向供应商发出请求if (m_order.checkStatus()) {m_nState++;m_vendor.submitRequest();std::cout << "********** 提交给供应商,跟踪次数:" <<  m_nCount << " **********" << std::endl;}break;case SubmittedToVendor:// 如果供应商已将包裹打包,将其转发给快递员if (m_vendor.checkStatus()) {m_nState++;m_courier.submitRequest();std::cout << "********** 提交给快递员,跟踪次数:" <<  m_nCount << " **********" << std::endl;}break;case SubmittedToCourier:// 如果包裹交付,订单完成if (m_courier.checkStatus())return true;default:break;}m_nCount++;// 订单未完成return false;}
private:enum States {Received,  // 收到SubmittedToOrderTeam,  // 提交给订单团队SubmittedToVendor,  // 提交给供应商SubmittedToCourier  // 提交给快递员};int m_nState;  // 订单状态int m_nCount;  // 跟踪次数OrderTeam m_order;Vendor m_vendor;Courier m_courier;
};#include "facade.h"
int main()
{OnlineShoppingFacade facade;// 提交订单facade.submitRequest();// 跟踪订单,直到订单完成while (!facade.checkStatus());std::cout << "********** 订单完成,跟踪次数:" << facade.followupNum() << "  **********" << std::endl;getchar();return 0;
}

 

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

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

相关文章

Sql注入(手工注入思路、绕过、防御)

一、Sql注入思路 1、判断注入点 在GET参数、POST参数、以及HTTP头部等&#xff0c;包括Cookie、Referer、XFF(X-Forwarded-for)、UA等地方尝试插入代码、符号或语句&#xff0c;尝试是否存在数据库参数读取行为&#xff0c;以及能否对其参数产生影响&#xff0c;如产生影响则…

thymeleaf学习

学习链接 th:fragment fragment的引用 th:insert:保留自己的主标签&#xff0c;保留th:fragment的主标签。 th:replace:不要自己的主标签&#xff0c;保留th:fragment的主标签。 th:include:保留自己的主标签&#xff0c;不要th:fragment的主标签。&#xff08;官方3.0后不推荐…

信创办公–基于WPS的EXCEL最佳实践系列 (数据整理复制粘贴)

信创办公–基于WPS的EXCEL最佳实践系列 &#xff08;数据整理复制粘贴&#xff09; 目录 应用背景操作步骤1、数据查找与替换2、复制或粘贴数据3、使用自动填充工具4、将数据拆分到多列5、应用数字格式 应用背景 数据的整理复制粘贴等在日常的工作中经常使用。本章内容主要学习…

第9章 Mybatis

9.1 谈谈你对Mybatis的理解 难度:★★ 重点:★★ 白话解析 说清楚Mybatis是什么,它的工作流程,然后再对比一下Hibernate就好了。 1、Mybatis是什么:它一个半自动ORM框架,它底层把JDBC那套加载驱动、创建连接、创建statement等重复性的硬编码全部给你封装好了,程序员只…

设计模式 - 享元模式

目录 一. 前言 二. 实现 一. 前言 享元模式&#xff08;Flyweight Pattern&#xff09;是一种结构型设计模式&#xff0c;它主要解决的问题是创建大量相似对象时的内存开销问题。该模式通过共享具有相同状态的对象来减少内存使用量。 享元模式的思想是&#xff1a;当需要创建…

嵌入式Linux应用开发-基础知识-第十九章驱动程序基石③

嵌入式Linux应用开发-基础知识-第十九章驱动程序基石③ 第十九章 驱动程序基石③19.5 定时器19.5.1 内核函数19.5.2 定时器时间单位19.5.3 使用定时器处理按键抖动19.5.4 现场编程、上机19.5.5 深入研究&#xff1a;定时器的内部机制19.5.6 深入研究&#xff1a;找到系统滴答 1…

SpringCloud(一)Eureka、Nacos、Feign、Gateway

文章目录 概述微服务技术对比 Eureka服务远程调用服务提供者和消费者Eureka注册中心搭建注册中心服务注册服务发现Ribbon负载均衡负载均衡策略饥饿加载 NacosNacos与Eureka对比Nacos服务注册Nacos服务分集群存储NacosRule负载均衡服务实例权重设置环境隔离 Nacos配置管理配置热…

ESP32设备驱动-OLED-SSD1306(I2C)显示屏驱动

OLED-SSD1306(I2C)显示屏驱动 1、OLED介绍 OLED显示屏是指有机电激发光二极管(OrganicLight-EmittingDiode,OLED)由于同时具备自发光,不需背光源、对比度高、厚度薄、视角广、反应速度快、可用于挠曲性面板、使用温度范围广、构造及制程较简单等优异之特性,被认为是下一…

C++面试题准备

文章目录 一、线程1.什么是进程&#xff0c;线程&#xff0c;彼此有什么区别?2.多进程、多线程的优缺点3.什么时候用进程&#xff0c;什么时候用线程4.多进程、多线程同步&#xff08;通讯&#xff09;的方法5.父进程、子进程的关系以及区别6.什么是进程上下文、中断上下文7.一…

短期风速预测|LSTM|ELM|批处理(matlab代码)

目录 1 主要内容 LSTM-长短时记忆 ELM-极限学习机 2 部分代码 3 程序结果 4 程序链接 1 主要内容 该程序是预测类的基础性代码&#xff0c;程序对河北某地区的气象数据进行详细统计&#xff0c;程序最终得到pm2.5的预测结果&#xff0c;通过更改数据很容易得到风速预测结…

Java 中的参数传递方式

Java 中的参数传递方式通常被称为“值传递”&#xff0c;这意味着在方法调用时&#xff0c;实际上传递给方法的是变量的副本&#xff0c;而不是变量本身。尽管这被广泛称为“值传递”&#xff0c;但需要注意的是&#xff0c;这并不意味着 Java 不支持引用传递。事实上&#xff…

WSL2安装历程

WLS2安装 1、系统检查 安装WSL2必须运行 Windows 10 版本 2004 及更高版本&#xff08;内部版本 19041 及更高版本&#xff09;或 Windows 11。 查看 Windows 版本及内部版本号&#xff0c;选择 Win R&#xff0c;然后键入winver。 2、家庭版升级企业版 下载HEU_KMS_Activ…

Django模板加载与响应

前言 Django 的模板系统将 Python 代码与 HTML 代码解耦&#xff0c;动态地生成 HTML 页面。Django 项目可以配置一个或多个模板引擎&#xff0c;但是通常使用 Django 的模板系统时&#xff0c;应该首先考虑其内置的后端 DTL&#xff08;Django Template Language&#xff0c;D…

【Flutter】Flutter Web 开发 如何从 URL 中获取参数值

【Flutter】Flutter Web 开发 如何从 URL 中获取参数值 文章目录 一、前言二、Flutter Web 中的 URL 处理三、如何从 URL 中获取参数四、实际业务中的用法五、完整示例六、总结 一、前言 大家好&#xff01;我是小雨青年&#xff0c;今天我想和大家分享一下在 Flutter Web 开发…

UGUI交互组件Button

一.初识Button对象 从菜单中创建Button对象&#xff0c;Button的文本由子节点Text对象显示&#xff0c;Button对象的组件除了基础组件外&#xff0c;还有Image用来显示Button常规态的图片&#xff0c;还有Button组件用来控制点击过渡效果和点击事件的响应。 二.Button组件的属…

C#,数值计算——Ranq1的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { /// <summary> /// Recommended generator for everyday use.The period is 1.8E19. Calling /// conventions same as Ran, above. /// </summary> public class Ranq1 { …

在亚马逊云科技控制台上创建 Amazon Cognito 用户池

介绍: Amazon Cognito 是一个强大的客户身份和访问管理服务&#xff0c;为您的面向客户的应用程序提供安全的认证和授权。本教程将指导您使用亚马逊云科技控制台创建 Amazon Cognito 用户池的过程。您将学习如何配置登录属性、安全要求、注册体验、消息传送和应用集成设置等各…

批量差异分析 批量findmarkers

各个单细胞亚群独立在两个分组做差异分析-腾讯云开发者社区-腾讯云 (tencent.com) Idents(sce) paste0(c,sce$group ) table(Idents(sce)) degs lapply(unique(sce$celltype), function(x){FindMarkers(sce[,sce$celltypex],ident.1 c1,ident.2 c2) }) xdegs[[1]] do.call…

Android 11.0 mt6771新增分区功能实现二

1.前言 在11.0的系统开发中,在对某些特殊模块中关于数据的存储方面等需要新增分区来保存, 所以就需要在系统分区新增分区,接下来就来实现这个功能,看系列二的实现过程 2.mt6771新增分区功能实现二的核心类 build/make/core/envsetup.mk build/make/core/main.mk build/m…

【2023年11月第四版教材】第17章《干系人管理》(第二部分)

第17章《干系人管理》&#xff08;第二部分&#xff09; 4 过程1-识别干系人4.1 数据收集★★★4.3数据分析4.4 权力利益方格4.5 数据表现&#xff1a;干系人映射分析和表现★★★ 5 过程2-规划干系人参与5.1 数据分析5.2 数据表现★★★5.2.1 干系人参与度评估矩阵★★★ 5.3 …