【设计模式深度剖析】【6】【结构型】【外观模式】| 以电脑开关按钮为例,并结合微服务架构的API网关加深理解

👈️上一篇:桥接模式

设计模式-专栏👈️

目 录

  • 外观模式(Facade Pattern)
  • 定义
    • 英文原文
    • 直译
    • 如何理解呢?
      • 字面理解
      • 代码实现中的理解
      • 生活案例:操作多功能料理机
      • 典型案例分析:API网关与外观模式
  • 外观模式角色
    • UML类图
    • 外观(Facade)
    • 子系统(Subsystem)
    • 客户(Client)
    • 代码示例
      • 子系统角色:SubsystemOne
      • 子系统角色:SubsystemTwo
      • 外观角色:Facade
      • 客户端角色:Client
  • 外观模式的应用
  • 外观模式的优点
  • 外观模式的缺点
  • 外观模式的使用场景
  • 示例解析:电脑开关按钮
    • 代码示例
      • UML类图
      • 子系统角色:电脑内部的复杂组件
      • 外观类(电源开关按钮)
      • 客户端

外观模式(Facade Pattern)

外观模式(Facade Pattern),如同一位精明的“外交官”,它简化了客户端与复杂子系统之间的交互,通过提供一个统一的“外交”接口,使得客户端无需深究子系统内部的“政治斗争”,从而降低了系统的复杂度,提高了代码的可维护性和可读性。

==>本文示例源码,点击查看👈️<==

定义

英文原文

Facade Pattern provides a unified interface to a set of interfaces in a subsystem. Facade defines a higher-level interface that makes the subsystem easier to use.

直译

外观模式为子系统中的一组接口提供了一个统一的接口。外观定义了一个更高层次的接口,使得子系统更加容易使用。

如何理解呢?

字面理解

外观即外表、外观面,是事物对外展示的一面。

代码实现中的理解

在代码实现中,外观角色就像是系统的“外表”,它隐藏了内部的子系统细节,为客户端提供了一个统一的接口来访问子系统。

生活案例:操作多功能料理机

想象一下,当我们正在使用一个复杂的厨房设备,比如多功能料理机。这个料理机内部可能包含了榨汁机、搅拌机、研磨机等多种功能。

但是,我们并不需要直接操作这些功能,只需要通过一个简单的控制面板就能完成我们想要的操作。

这个控制面板就是外观模式的一个实际应用。它隐藏了料理机内部的复杂性,为我们提供了一个简单、直观的操作方式。

同样地,在软件系统中,外观模式也为我们隐藏了复杂的内部结构和交互,使得我们能够更加方便地使用系统。

典型案例分析:API网关与外观模式

微服务架构中的API网关,为客户端提供了统一的访问入口,是进入系统的唯一节点,这与外观模式的定义相符。API网关封装了内部系统的架构,并且提供API给各个客户端。

网关处理非业务逻辑(如认证、授权、负载均衡、缓存、熔断、降级、限流等),使得业务微服务更加专注于业务实现。

具体表现为:

  1. 隐藏后端服务的复杂性

    在微服务架构中,后端通常包含多个独立的服务,每个服务都有自己的接口、功能和数据。这些服务之间可能存在复杂的依赖关系和交互逻辑。API网关作为一个中间件,隐藏了这些复杂性,为前端提供了一个统一、简洁的接口。前端开发者无需为每一个后端微服务单独配置地址和端口,只需要知道API网关的地址即可。因此,前端开发者无需关心后端服务的具体实现和部署情况,只需与API网关进行交互即可。

  2. 服务聚合

    当一个前端请求需要多个后端微服务协同处理时,API网关可以聚合这些微服务的响应,并将结果合并后返回给前端。

    API网关能够隐藏后端服务的复杂性,这减少了前端的复杂性,为前端提供一个简洁的响应,使得前端能够专注于与用户的交互和展示逻辑。

  3. 统一接口和协议

    API网关为前端提供了统一的接口和协议,使得前端开发者能够以统一的方式与后端服务进行通信。这样,即使后端服务的接口发生变化或新增了服务,只要API网关的接口保持不变,前端代码就无需修改。这种统一性减少了前端与后端之间的耦合度,使得前端可以更加灵活地与后端进行交互。

  4. 认证和授权

    API网关可以实施身份验证和授权机制,确保只有合法的用户、经过认证的请求才能访问后端服务。

    它可以通过OAuth2.0、JWT等技术实现用户认证和令牌管理,保护后端服务免受非法访问。API网关成为安全性的守护者,为后端服务提供一层保护。

  5. 限流和熔断

    API网关可以对流量进行限制,防止后端服务因过载而崩溃。

    当某个微服务出现故障时,API网关可以自动熔断对该微服务的调用,避免故障扩散到其他服务,避免级联故障的发生。

    API网关能够控制和管理后端服务的访问,确保系统的稳定性和可靠性。

  6. 路由和请求转发

    API网关能够根据预设的路由规则,将前端的请求转发到后端对应的服务上。这种路由机制使得前端无需关心后端服务的具体地址和端口,只需发送请求到API网关即可。同时,API网关还可以根据服务的负载情况、健康状况等因素进行负载均衡和容错处理,进一步降低前端与后端之间的耦合度。

  7. 负载均衡

    一个网关可以接收多个服务实例,因此网关需要在各个对等的服务实例上做负载均衡策略。

  8. 降低服务间的耦合度

    通过将非业务逻辑(如路由、转发、认证、授权、熔断、限流等)放在API网关中实现,可以使得后台业务微服务更加专注于业务逻辑的实现,实现业务与技术的解耦。

    通过API网关的路由转发机制,前端与后端之间的耦合度被大大降低。前端开发者无需关心后端服务的具体实现和部署情况,只需与API网关进行交互即可。同时,后端服务无需直接关注前端发送的具体请求细节,包括请求的参数、格式等,同时也无需直接处理针对前端的响应格式。它们只需按照与API网关或其他服务调用者约定的接口规范,处理接收到的请求数据,并返回符合规范的数据格式。

  9. 日志和监控

    API网关可以记录所有经过的请求日志,方便后续的故障排查、性能分析和安全审计。

    收集API的调用量、响应时间、错误率等关键指标,为系统监控和性能调优提供数据支持。

通过外观模式的实现,API网关在前后端分离的微服务架构中发挥着至关重要的作用。它简化了前端的配置和开发过程,提高了系统的可扩展性、可靠性和安全性。同时,API网关还提供了路由、认证、限流、监控等核心功能,使得微服务架构更加完善和高效。

外观模式角色

UML类图

在这里插入图片描述

外观(Facade)

为子系统中的一组接口提供一个一致的界面,此界面使得这一子系统更加容易使用。

子系统(Subsystem)

实现系统的部分功能,客户可以通过外观角色与其进行交互。

客户(Client)

通过外观角色与子系统交互,使用子系统的功能。

代码示例

子系统角色:SubsystemOne

package com.polaris.designpattern.list2.structural.pattern6.facade.classicdemo;

// 子系统角色:SubsystemOne
public class SubsystemOne {
    public void methodOne() {
        System.out.println("SubsystemOne: Method 1 executed.");
    }
}  

子系统角色:SubsystemTwo

package com.polaris.designpattern.list2.structural.pattern6.facade.classicdemo;

// 子系统角色:SubsystemTwo  
public class SubsystemTwo {
    public void methodTwo() {
        System.out.println("SubsystemTwo: Method 2 executed.");
    }
}  

外观角色:Facade

package com.polaris.designpattern.list2.structural.pattern6.facade.classicdemo;

// 外观角色:Facade  
public class Facade {
    private SubsystemOne subsystemOne;
    private SubsystemTwo subsystemTwo;

    public Facade() {
        subsystemOne = new SubsystemOne();
        subsystemTwo = new SubsystemTwo();
    }

    // 提供单独调用子系统一的方法  
    public void operationOne() {
        subsystemOne.methodOne();
    }

    // 提供单独调用子系统二的方法  
    public void operationTwo() {
        subsystemTwo.methodTwo();
    }

    // 提供同时调用两个子系统方法的方法  
    public void operation() {
        operationOne();
        operationTwo();
    }
}  

客户端角色:Client

package com.polaris.designpattern.list2.structural.pattern6.facade.classicdemo;


// 客户端角色:Client  
public class Client {
    public static void main(String[] args) {
        Facade facade = new Facade();

        // 单独调用子系统一的方法  
        facade.operationOne();
        System.out.println("------------------");

        // 单独调用子系统二的方法  
        facade.operationTwo();
        System.out.println("------------------");

        // 同时调用两个子系统的方法  
        facade.operation();
    }
}
/* Output:
SubsystemOne: Method 1 executed.
------------------
SubsystemTwo: Method 2 executed.
------------------
SubsystemOne: Method 1 executed.
SubsystemTwo: Method 2 executed.
*///~

外观模式的应用

外观模式就像是我们日常生活中使用的智能手机界面。手机内部可能包含了各种复杂的硬件和软件,但对外,我们只需要通过一个简单、直观的界面就能完成打电话、发信息、拍照、上网等各种功能。这个界面就是外观模式的一个应用,它隐藏了内部的复杂性,为用户提供了一个统一的、简单的操作方式。

外观模式的优点

  1. 简化操作:外观模式为用户提供了一个统一的接口,用户无需了解内部复杂的结构,只需通过外观接口就能完成所需的操作。
  2. 降低耦合度:外观模式使得子系统之间的耦合度降低,因为它们之间的交互都是通过外观接口进行的,而不是直接交互。
  3. 提高灵活性:当子系统内部发生变化时,只需要修改外观接口的实现,而无需修改客户端代码,提高了系统的灵活性。

外观模式的缺点

  1. 增加复杂性:在某些情况下,如果设计不当,外观类可能会变得非常复杂,从而增加了系统的复杂性。
  2. 依赖风险:由于客户端都依赖于外观接口,因此如果外观接口设计不合理或出现问题,可能会导致整个系统的崩溃。

外观模式的使用场景

  1. 子系统之间交互复杂:当系统内部有多个子系统,并且子系统之间的交互非常复杂时,可以使用外观模式来简化它们之间的交互。
  2. 需要隐藏子系统细节:在某些情况下,我们可能不希望用户或客户端直接访问子系统的内部细节。此时,可以使用外观模式来隐藏这些细节,只提供一个统一的接口供用户访问。
  3. 提高系统灵活性和可扩展性:当系统需要经常添加或删除子系统时,使用外观模式可以使得客户端代码不随子系统的变化而变化,从而提高了系统的灵活性和可扩展性。

示例解析:电脑开关按钮

==>本文示例源码,点击查看👈️<==

我们可以使用“电脑开关机”作为一个更简单且常见的外观模式例子。

电脑内部有很多复杂的硬件和软件组件,如处理器、内存、硬盘、操作系统等。但是,当我们想要打开电脑或关闭电脑时,我们并不需要直接与这些组件交互,而是通过一个统一的接口——电源开关按钮来完成。

我们并不需要了解电脑内部复杂的构造和工作原理,只需要按下电源开关按钮,电脑就会开始启动;再次按下按钮,电脑就会关机。

电源开关按钮就是外观模式的一个实际应用,它隐藏了电脑内部复杂的启动和关闭过程,为用户提供了一个简单、直观的操作方式。

代码示例

UML类图

在这里插入图片描述

子系统角色:电脑内部的复杂组件

package com.polaris.designpattern.list2.structural.pattern6.facade.computerfacade;

// 电脑内部的复杂组件(这里只是示意)
public class ComputerHardware {
    public void startUp() {
        System.out.println("Starting up the computer...");
        // 这里会执行一系列复杂的启动过程,如加载操作系统、初始化硬件等  
    }

    public void shutDown() {
        System.out.println("Shutting down the computer...");
        // 这里会执行一系列复杂的关闭过程,如保存数据、关闭应用程序、关闭硬件等  
    }
}  

外观类(电源开关按钮)

package com.polaris.designpattern.list2.structural.pattern6.facade.computerfacade;

// 外观类(电源开关按钮)
public class PowerButton {
    private ComputerHardware hardware;

    public PowerButton() {
        hardware = new ComputerHardware();
    }

    // 统一的接口,用户只需调用此方法即可启动电脑  
    public void pressToStart() {
        hardware.startUp();
    }

    // 统一的接口,用户只需调用此方法即可关闭电脑  
    public void pressToShutDown() {
        hardware.shutDown();
    }
}  
  

客户端

package com.polaris.designpattern.list2.structural.pattern6.facade.computerfacade;


// 客户端代码(用户)  
public class User {
    public static void main(String[] args) {
        PowerButton powerButton = new PowerButton();
        powerButton.pressToStart(); // 启动电脑  
        // ... 用户使用电脑 ...  
        powerButton.pressToShutDown(); // 关闭电脑  
    }
}

/* Output:
Starting up the computer...
Shutting down the computer...
*///~

在这个代码示例中,

ComputerHardware 类代表电脑内部的复杂组件,它提供了启动和关闭电脑的方法。

PowerButton 类是外观类,它隐藏了电脑内部复杂的启动和关闭过程,并为用户提供了一个统一的接口——电源开关按钮。

用户(即客户端)只需要与 PowerButton 类交互,就能控制电脑的启动和关闭,而无需关心电脑内部是如何工作的。

👈️上一篇:桥接模式

设计模式-专栏👈️

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【蓝桥杯】第十四届蓝桥杯大赛软件赛国赛C/C++ 大学 B 组

答题结果页 - 蓝桥云课 (lanqiao.cn) 0子2023 - 蓝桥云课 (lanqiao.cn)&#xff08;暴力枚举 #include<bits/stdc.h> using lllong long; using ullunsigned long long; #define fir first #define sec second //#define int llconst int N1e510; const int mod1e97;int…

多线程新手村4--定时器

定时器是日常开发中很常见的组件&#xff0c;定时器大家可能不知道是干什么的&#xff0c;但是定时炸弹肯定都听过&#xff0c;定个时间&#xff0c;过一段时间后bomb&#xff01;&#xff01;&#xff01;爆炸 定时器的逻辑和这个一样&#xff0c;约定一个时间&#xff0c;这…

windows ollama 指定模型下载路径

为Ollama指定模型的下载路径 在Windows系统中&#xff0c;如果想为Ollama指定模型的下载路径&#xff0c;可以通过设置环境变量来实现。以下是详细的步骤&#xff1a; 确定默认下载路径&#xff1a; 默认情况下&#xff0c;Ollama的模型可能会下载到C:\Users\<用户名>…

类别型特征

#机器学习 #深度学习 #基础知识 #特征工程 #数据编码 背景 在现实生活中,我们面对的数据类型有很多,其中有的数据天然为数值类型具备数值意义,那么可以很自然地和算法结合,但是大部分数据他没有天然的数值意义,那么将他们送入到算法前,就需要对数据进行编码处理,将其转换为数…

如何成为快手外卖代理?本地生活服务平台加盟条件解析

近年来&#xff0c;以抖音、快手和小红书等为代表的互联网大厂纷纷进军本地生活领域&#xff0c;改变美团和饿了么二分天下的这一局面的同时&#xff0c;也让本地生活成为了众多创业者眼中的“香饽饽”。其中&#xff0c;快手凭借着其庞大的用户群体&#xff0c;让快手团购外卖…

js使用链表实现音乐播放器(新增,下一首播放,置顶,删除)

什么是链表 链表是一种线性数据结构&#xff0c;与数组类似&#xff0c;它用于存储一系列元素。不过&#xff0c;与数组在内存中连续存储元素不同&#xff0c;链表中的元素&#xff08;称为节点&#xff09;在内存中可以是非连续存放的。每个节点包含两部分&#xff1a;一部分…

利用CNN和迁移学习检测太阳能电池板上的尘埃

太阳能电池板已经成为农业、交通、建筑和酒店等多个行业中受欢迎的可再生能源来源。通过利用太阳的能量&#xff0c;我们可以在不损害环境的情况下产生电力。然而&#xff0c;使用太阳能电池板面临着一些挑战&#xff0c;其中最大的之一是它们表面上尘埃的积累。这会显著降低它…

【机器学习】MS_MARCO_Web_Search解析说明

MS MARCO Web Search&#xff1a;引领大型模型与信息检索的新纪元 一、引言&#xff1a;大型模型与信息检索的挑战二、MS MARCO Web Search数据集的特点三、MS MARCO Web Search数据集的应用五、结语 在信息爆炸的时代&#xff0c;如何高效、准确地从海量数据中检索出有价值的信…

Java SE基础知识(13)

知识梳理&#xff1a; package LocalDate;import java.time.DayOfWeek; import java.time.LocalDate; import java.time.Month; import java.time.MonthDay; import java.util.Locale;public class demo1 {public static void main(String[] args) {//1.获取当前时间的日历对象…

构建php环境、安装、依赖、nginx配置、ab压力测试命令、添加php-fpm为系统服务

目录 php简介 官网php安装包 选择下载稳定版本 &#xff08;建议使用此版本&#xff0c;文章以此版本为例&#xff09; 安装php解析环境 准备工作 安装依赖 zlib-devel 和 libxml2-devel包。 安装扩展工具库 安装 libmcrypt 安装 mhash 安装mcrypt 安装php 选项含…

VBA技术资料MF158:获取系统的用户名

我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高自己的工作效率&#xff0c;而且可以提高数据的准确度。“VBA语言専攻”提供的教程一共九套&#xff0c;分为初级、中级、高级三大部分&#xff0c;教程是对VBA的系统讲解&#…

GBB和Prob IoU[旋转目标检测理论篇]

在开始介绍YOLOv8_obb网络之前,需要先介绍一下arxiv.org/pdf/2106.06072 这篇文章的工作,因为v8_obb就是基于这篇论文提出的GBB和prob IoU来实现旋转目标检测的。 1.高斯分布 一维高斯分布的规律是中间高两边低,且当x为均值的时候取到最大值,表达式如下,标准正态分布图如…

加密资产私钥安全完整手册(一) ,bitget钱包为例

比特币和以太坊等加密货币的兴起开创了数字金融的新时代&#xff0c;但也带来了独特的安全挑战。这些代表现实世界价值的数字资产已成为黑客和窃贼的主要目标。为了安全地应对这种情况&#xff0c;了解私钥的基本概念至关重要。 私钥是加密货币所有权和安全性的基石。它们相当于…

基于图卷积网络的人体3D网格分割

深度学习在 2D 视觉识别任务上取得了巨大成功。十年前被认为极其困难的图像分类和分割等任务&#xff0c;现在可以通过具有类似人类性能的神经网络来解决。这一成功归功于卷积神经网络 (CNN)&#xff0c;它取代了手工制作的描述符。 NSDT工具推荐&#xff1a; Three.js AI纹理开…

阿里云和AWS的CDN产品对比分析

在现代互联网时代,内容分发网络(CDN)已成为确保网站和应用程序高性能和可用性的关键基础设施。作为两家领先的云服务提供商,阿里云和Amazon Web Services(AWS)都提供了成熟的CDN解决方案,帮助企业优化网络传输和提升用户体验。我们九河云一直致力于阿里云和AWS云相关业务&#…

【全开源】西陆家政系统源码小程序(FastAdmin+ThinkPHP+原生微信小程序)

打造高效便捷的家政服务平台 一、引言&#xff1a;家政服务的数字化转型 随着人们生活节奏的加快&#xff0c;家政服务需求日益增长。为了满足广大用户对高效、便捷的家政服务的需求&#xff0c;家政小程序系统源码应运而生。这款源码不仅能够帮助家政服务提供商快速搭建自己…

IDEA2023.2单击Setting提示报错:Cannot get children Easy Code

1、单击Setting&#xff0c;不能弹出对话框 2、打开IDE Internal Errors发生错误 原因&#xff1a; 报错信息 "Cannot get children Easy Code" 通常指的是 IntelliJ IDEA 在尝试访问或操作 Easy Code 插件的子设置时遇到了问题。 主要检查是有网络&#xff08;断断…

R包Colorfindr识别图片颜色|用刀剑神域方式打开SCI科研配色

1.前言 最近忙里偷闲&#xff0c;捣鼓一下配色&#xff0c;把童年回忆里的动漫都搬进来&#xff0c;给科研信仰充值吧&#xff5e; 提取颜色之前写过一个Py的&#xff0c;那个很准确不过调参会有点麻烦。这里分享一个比较懒人点的R包吧&#xff0c;虽然会有一定误差&#xff…

【XR806开发板试用】基础篇,从零开始搭建一个LCD彩屏时钟(ST7735S驱动)

本文从搭建环境开始&#xff0c;step by step教大家使用XR806实现驱动SPI屏幕&#xff08;ST7735S驱动&#xff09;&#xff0c;并连接WiFi实现ntp对时&#xff0c;最终实现把时间显示到屏幕上。 #1. 搭建开发环境 1. 安装编译环境所需的依赖包 基于ubuntu 20.04&#xff0c;按…

作业-day-240522

思维导图 使用IO多路复用实现并发 select实现TCP服务器端 #include <myhead.h>#define SER_IP "192.168.125.112" #define SER_PORT 8888int main(int argc, const char *argv[]) {int sfdsocket(AF_INET,SOCK_STREAM,0);if(sfd -1){perror("socket er…