当前位置: 首页 > news >正文

设计模式⑦ :简单化

文章目录

  • 一、前言
  • 二、Facade 模式
    • 1. 介绍
    • 2. 应用
    • 3. 总结
  • 三、Mediator 模式
    • 1. 介绍
    • 2. 应用
    • 3. 总结

一、前言

有时候不想动脑子,就懒得看源码又不像浪费时间所以会看看书,但是又记不住,所以决定开始写"抄书"系列。本系列大部分内容都是来源于《 图解设计模式》(【日】结城浩 著)。该系列文章可随意转载。

二、Facade 模式

Facade 模式 :简单窗口

1. 介绍

一般来说,随着时间推移,程序会变得越来越复杂,这使得程序结构也变得越来越复杂,在使用这些功能时,我们需要先整理清楚他们之间的关系,注意正确的调用顺序。与其如此,不如为这个大型程序准备一个“窗口“,这样就无需关注每个类,只需要简单的对窗口提出请求即可。

使用 Facade 模式可以为互相关联在一起的错综复杂的类整理出高层接口。其中 Facade 角色可以让系统对外只有一个简单的接口。而且 Facade 角色还会考虑到系统内部各个类之间的责任关系和依赖关系,按照正确的顺序调用各个类。


Facade 模式 中出场的角色

  • Facade(窗口): 该角色是代表构成系统的许多其他角色的简单窗口。Facade 角色向系统外部提供高层接口
  • 构成系统的许多其他角色 :这些角色各自完成自己的工作,他们并不知道Facade 角色。Facade 角色调用其他角色进行工作,但是其他角色并不会调用 Facade角色。
  • Client(请求者):该角色负责调用 Facade 角色。

类图如下:

在这里插入图片描述


Demo如下:

// 基础数据获取
public class Database {/*** 防止外部创建*/private Database() {}/*** 获取基础数据* @return*/@SneakyThrowspublic static Properties getProperties() {Properties prop = new Properties();prop.setProperty("123456789@qq.com", "张三");prop.setProperty("987654321@qq.com", "李四");return prop;}
}// html 工具类
public class HtmlWriter {/*** 生成Html内容** @param title* @throws IOException*/public static String createContent(String title, String userName, String email) throws IOException {return "<html>" +"<head>" +"<title>" + title + "</title>" +"</head>" +"<body>" +"<h1>" + title + "</h1>" +"<div> 欢迎: " + userName + "</div>" +"<a href=\"" + email + "\"> " + email + "</a>" +"</html>";}/*** 生成html文件** @param fileName* @param content*/public static void writeHtml(String fileName, String content) {FileUtil.writeBytes(content.getBytes(StandardCharsets.UTF_8), fileName);}}// 页面生成类,
public class PageMaker {private PageMaker(){}/*** 构建欢迎页面*/public static void markWelcomePage(String mailaddr) throws IOException {final Properties prop = Database.getProperties();final String userName = prop.getProperty(mailaddr);final String content = HtmlWriter.createContent(userName, userName, mailaddr);HtmlWriter.writeHtml("D://welcome.html", content);}
}public class FacadeDemoMain {public static void main(String[] args) throws IOException {//调用 PageMaker#markWelcomePage 生成 html 页面PageMaker.markWelcomePage("123456789@qq.com");}
}

结果如下:生成welcome.html 文件,打开内容如图
在这里插入图片描述

对用户来说,只需要调用 PageMaker#markWelcomePage,传入指定邮箱便可以生成对应的 HTML页面,而其内部实现的HtmlWriter和Database对用户来说是无法感知的,即 PageMaker 对于 生成 Html 这个功能来说就是一个 Facade。

2. 应用

  • Dubbo 的分层模式个人认为也是Facade 模式的应用:Dubbo提供各个SPI 接口,如果用户需要扩展或改变功能只需要重新对应的SPI 即可。而无需关心 SPI 的上下游的关联和调用关系。如下图:
    在这里插入图片描述

  • Facade 模式在项目中几乎无时无刻不在用到,这里以 Spring 中随便找的一个方法为例:DispatcherServlet#doDispatch 中 会调用 HandlerExecutionChain#applyPreHandle 方法执行拦截器方法,那么这里就可以认为 HandlerExecutionChain 对 HandlerInterceptor 进行了分装。DispatcherServlet 只需要调用HandlerExecutionChain#applyPreHandle 便可以执行拦截器方法而无需关注具体实现。

在这里插入图片描述



个人使用:该部分内容是写给自己看的,帮助自身理解,因此就不交代项目背景了,读者请自行忽略(◐ˍ◑)

  • 个人理解在微服务级别,网关也起到类似 Facade 的作用:网关对外部暴露的接口是有限且可控的,外部无需关注网关内部接口的调用关系。

  • 在项目A 中,对同一数据的获取由于各个地区不同获取的途径或调用的第三方接口都不相同,此时需要在项目中对该数据的获取封装出一个 Facade ,当需要获取数据时只需要调用 Facade 暴露出的接口即可。Demo 如下:

    public interface DataSource {/*** 获取数据* @return*/String getData();
    }public class HttpDataSource implements DataSource {@Overridepublic String getData() {return "这是从 Http 调用获取的数据";}
    }public class DataBaseDataSource implements DataSource {@Overridepublic String getData() {return "这是从数据库获取的数据";}
    }public class DataFacade {private DataBaseDataSource dataBaseDataSource = new DataBaseDataSource();private HttpDataSource httpDataSource = new HttpDataSource();/*** 获取数据,nb 从数据库获取,sh通过http调用获取* @param region* @return*/public String getData(String region) {if ("nb".equals(region)) {return dataBaseDataSource.getData();} else if ("sh".equals(region)) {return httpDataSource.getData();} else {throw new RuntimeException("不支持");}}}public class DemoMain {public static void main(String[] args) {DataFacade dataFacade = new DataFacade();System.out.println(dataFacade.getData("nb"));System.out.println(dataFacade.getData("sh"));}
    }

    输出如下:

    在这里插入图片描述

3. 总结

扩展思路:

  • Facade 角色的工作:Facade 模式可以让复杂的东西看起来简单,使用 Facade 模式可以让我们不再在意后台工作的类和他们之间的关系。这里的重点是接口的变少了,这意味着程序和外部的关联关系弱化了,这使得我们的包作为组件更方便被复用。
  • 递归的使用Facade模式:在超大系统中往往包含非常多的包和类,如果我们在每个关键的地方都使用Facade 模式,系统的维护就会变的简单。

相关设计模式:

  • Abstract Factory 模式:可以将 Abstract Factory 模式看做是生成复杂实例时的 Facade 模式。因为它提供了“要想生成这个实例只需要调用这个方法就ok了”的简单接口。
  • Singleton 模式:有时会使用 Singleton 模式创建 Facade 角色。
  • Mediator 模式:在 Mediator 模式汇总, Mediator 角色与 Colleague 角色进行交互。而在 Facade 模式中,Facade 角色单方面地使用其他角色来对外提供高层接口。因此可以说 Mediator 是双向的,而 Facade 是单向的。

一时的小想法,仅仅个人理解,无需在意 :

  • Facade 模式不仅仅可以用于项目的代码中,在微服务搭建时,网关或者交换中心也起到类似的作用。对于外部访问,暴露出有限且可控的接口。
  • 个人认为最为简单的应用就是 MVC 中 Service 层暴露出的接口,每个 Service 接口内部需要调用多个 Dao 层的接口才能实现,那么就可以认为 Service 层是 Dao 层的封装。区别就是这里的 Service 层的“窗口”粒度更细。

三、Mediator 模式

Mediator 模式 : 只有一个仲裁者。

1. 介绍

当麻烦事发生时通知仲裁者,当发生涉及全体组员的事情,也通知仲裁者,当仲裁者下达指示时,组员会立即执行。组员之间不再相互沟通作出决定,而是发生任何事都向仲裁者报,一方面仲裁者站在整个团队的角度上对组员上报的事情做出决定。最后,整个团队的交流过程就变为了组员向仲裁者报告,仲裁者向组员下达指示。


Mediator 中出场的角色

  • Mediator(仲裁者、中介者):该角色负责定义与 Colleague 角色进行通信和做出决定的接口(API)
  • ConcreteMediator(具体的仲裁者、中介者):该角色负责实现 Mediator 角色的接口,负责作出实际决定。
  • Colleague(同事):该角色负责定义与 Mediator 角色进行通信的接口。
  • ConcreteColleague(具体的同事):该角色负责实现 Colleague 角色的接口。

类图如下:

在这里插入图片描述


2. 应用

  • Mediator 模式非常典型的应用则是在集群服务异常下线时的选举,如Redis的哨兵模式的监视和选举。
    以下面为例(内容来自于书籍 《Redis设计与实现》):

    1. 如下图展示了一个哨兵系统,server1 为主服务器,server2,server3, server4 为从服务器,sentinel 系统监视着这四个服务器
      在这里插入图片描述
    2. 假设这时server1 进入下线状态,那么从服务器 server2,server3, server4 对主服务器的复制将会被终中止,并且 Sentinel 系统会查询到 server1下线

    在这里插入图片描述
    3. 当 server1 的下线时长超过用户设定的下线时长上限时,Sentinel 系统就会对 server1 执行故障转移操作:

    1. Sentinel 系统会挑选 server1 的从服务器中的一个升级为主服务器
    2. 之后Sentinel 系统会想 server1 的所有从服务器发送新的复制指令,让他们成为新的主服务器的从服务器,当所有从服务器都开始复制新的主服务器时,故障转移操作执行完毕。
      另外,Sentinel 还会监视已经下线的 server1,当其重新上线时会将其设置为新的主服务器的从服务器。
      在这里插入图片描述


个人使用:该部分内容是写给自己看的,帮助自身理解,因此就不交代项目背景了,读者请自行忽略(◐ˍ◑)

  • Mediator 模式在目前的经验中似乎并没有用到,不过临时想到了一个应用场景:类如支付宝登录这种需要极度安全的操作肯定不能直接通过用户名密码验证,需要根据当前使用的手机是否新手机、当前账号多久没有登录、截止目前登录了多少次等种种因素决定当前登录是否有效,如此便可以使用仲裁者模式。Demo如下:

    // 判断接口
    public interface Colleague {/*** 信息判决* @param userInfo* @return*/int judgment(String userInfo);
    }public class LoginMadiator {// 登录次数验证private NumberColleague numberColleague = new NumberColleague();// 手机号验证private PhoneColleague phoneColleague = new PhoneColleague();/*** 登录判决** @return*/public boolean loginJudgment(String userInfo) {// 根据用户信息获取登录次数的判定final int numberScore = numberColleague.judgment(userInfo);// 根据用户信息获取是否新手机的的判定final int phoneScore = phoneColleague.judgment(userInfo);// 如果登录次数或是否新手机的判定有一个小于80分,本次登录验证不通过。return numberScore < 80 || phoneScore < 80;}
    }public class DemoMain {public static void main(String[] args) {String userName = "夏义春";String password = "123456789";LoginService loginService = new LoginService ();// 用户登录, 返回用户登录信息final String userInfo = loginService.login(userName, password);LoginMadiator loginMadiator = new LoginMadiator();// 如果本次登录裁决失败,则进行人脸或短信验证if (!loginMadiator.loginJudgment(userInfo)){// TODO : 进行人脸验证 或短信验证}}
    }
    

3. 总结

相关的设计模式:

  • Facade 模式:在 Mediator 模式汇总, Mediator 角色与 Colleague 角色进行交互。而在 Facade 模式中,Facade 角色单方面地使用其他角色来对外提供高层接口。因此可以说 Mediator 是双向的,而 Facade 是单向的
  • Observer 模式:有时会使用Observer 模式来实现 Mediator 角色与 Colleague 角色之间的通信。

一时的小想法,仅仅个人理解,无需在意 :

  • 一个仲裁类控制其他类来完成任务,也可以存在变种:即仲裁者也可以变成组员,类似 主从模式,可以自由选举。总的来说,需要抉择多重的任务级别适用于仲裁者模式。

相关文章:

设计模式⑦ :简单化

文章目录 一、前言二、Facade 模式1. 介绍2. 应用3. 总结 三、Mediator 模式1. 介绍2. 应用3. 总结 一、前言 有时候不想动脑子&#xff0c;就懒得看源码又不像浪费时间所以会看看书&#xff0c;但是又记不住&#xff0c;所以决定开始写"抄书"系列。本系列大部分内容…...

Java:选择哪个Java IDE好?

Java&#xff1a;选择哪个Java IDE好? 在开始前我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「java的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&…...

unity打包apk后网络请求提示unknown error处理

近期同事的一个比较老的版本的unity项目在电脑上运行都正常&#xff0c;但是打包成android后安装到手机上就提示unknown error 让我帮他排查一下问题。由于是涉密项目不能发图就简单介绍下处理过程吧&#xff01; 一、故障原因 请求的地址ssl证书过期了。 二、处理过程 更改请…...

力扣 | 11. 盛最多水的容器

双指针解法–对撞指针 暴力解法public int maxArea1(int[] height) {int n height.length;int ans 0;for (int i 0; i < n; i) {for (int j i 1; j < n; j) {int area Math.min(height[i], height[j]) * (j - i);ans Math.max(ans, area);}}return ans;}双指针解法…...

史上最全EasyExcel

一、EasyExcel介绍 1、数据导入&#xff1a;减轻录入工作量 2、数据导出&#xff1a;统计信息归档 3、数据传输&#xff1a;异构系统之间数据传输 二、EasyExcel特点 Java领域解析、生成Excel比较有名的框架有Apache poi、jxl等。但他们都存在一个严重的问题就是非常的耗内…...

MySQL---事务的四大特性详解(高频面试题)

在MySQL中&#xff0c;事务具有以下四个基本特性&#xff1a; 原子性&#xff08;Atomicity&#xff09;、一致性&#xff08;Consistency&#xff09;、隔离性&#xff08;Isolation&#xff09;和持久性&#xff08;Durability&#xff09;。这些特性通常被称为ACID特性。 一…...

为 OpenCV 编写文档(二)

常用命令 这里通过简短的示例描述了最常用的 doxygen 命令。有关可用命令的完整列表和详细说明&#xff0c;请访问命令参考。 基本命令 brief - 带有简要实体描述的段落 param - 函数参数的描述。 多个相邻语句合并到一个列表中。如果在实际函数签名中找不到具有此名称的参数…...

HUAWEI华为MateStation S台式机电脑12代PUC-H7621N,H5621N原装出厂Windows11.22H2系统

链接&#xff1a;https://pan.baidu.com/s/1QtjLyGTwMZgYiBO5bUVPYg?pwd8mx0 提取码&#xff1a;8mx0 原厂WIN11系统自带所有驱动、出厂主题壁纸、系统属性专属联机支持标志、Office办公软件、华为电脑管家等预装程序 文件格式&#xff1a;esd/wim/swm 安装方式&#xf…...

机器学习:holdout法(Python)

import pandas as pd import numpy as np from sklearn.preprocessing import LabelEncoder, StandardScaler # 类别标签编码&#xff0c;标准化处理 from sklearn.decomposition import PCA # 主成分分析 import matplotlib.pyplot as plt from sklearn.model_selection impor…...

【GaussDB数据库】序

参考链接1&#xff1a;国产数据库华为高斯数据库&#xff08;GaussDB&#xff09;功能与特点总结 参考链接2&#xff1a;GaussDB(DWS)介绍 GaussDB简介 官方网站&#xff1a;云数据库GaussDB GaussDB是华为自主创新研发的分布式关系型数据库。该产品支持分布式事务&#xff0c;…...

代码随想录算法训练营第三十八天|理论基础、509. 斐波那契数、70. 爬楼梯、746. 使用最小花费爬楼梯

题目&#xff1a;理论基础 文章链接&#xff1a;代码随想录 视频链接&#xff1a;动态规划理论基础 动态规划五部曲&#xff1a; 确定dp数组&#xff08;dp table&#xff09;以及下标的含义确定递推公式dp数组如何初始化确定遍历顺序举例推导dp数组 题目&#xff1a;509. 斐…...

大数据开发之Hadoop(优化新特征)

第 1 章&#xff1a;HDFS-故障排除 注意&#xff1a;采用三台服务器即可&#xff0c;恢复到Yarn开始的服务器快照。 1.1 集群安全模块 1、安全模式&#xff1a;文件系统只接收读数据请求&#xff0c;而不接收删除、修改等变更请求 2、进入安全模式场景 1&#xff09;NameNod…...

在使用go语言开发的时候,程序启动后如何获取程序pid

在Go语言中&#xff0c;标准库并没有直接提供获取进程ID&#xff08;PID&#xff09;的函数。通常&#xff0c;你可以使用os包和syscall包来调用底层的操作系统函数来获取PID。 以下是一个获取程序PID的示例代码&#xff1a; package mainimport ("fmt""os&qu…...

HFSS笔记/信号完整性分析(二)——软件仿真设置大全

文章目录 1、多核运算设置1.1 如何设置1.2 如何查看自己电脑的core呢&#xff1f;1.3 查看求解的频点 2、求解模式设置Driven Terminal vs Driven modal 3、Design settings4、自适应网格划分5、更改字体设置 仅做笔记整理与分享。 1、多核运算设置 多核运算只对扫频才有效果&…...

mysql主从报错:Last_IO_Error: Error connecting to source解决方法

目录 报错 处理方法 1.从库停止同步 2.主库修改my.cnf 生效配置default-authentication-pluginmysql_native_password 3.重启服务重新创建复制用户 4.重新同步 5.测试主从 报错 Last_IO_Error: Error connecting to source repl_user192.168.213.15:3306. This was atte…...

AOI与AVI:在视觉检测中的不同点和相似点

AOI&#xff08;关注区域&#xff09;和AVI&#xff08;视觉感兴趣区域&#xff09;是视觉检测中常用的两个概念&#xff0c;主要用于识别和分析图像或视频中的特定区域。虽然这两个概念都涉及到注视行为和注意力分配&#xff0c;但它们在定义和实际应用等方面有一些差异。 AOI…...

Python爬虫 - 网易云音乐下载

爬取网易云音乐实战&#xff0c;仅供学习&#xff0c;不可商用&#xff0c;出现问题&#xff0c;概不负责&#xff01; 分为爬取网易云歌单和排行榜单两部分。 因为网页中&#xff0c;只能显示出歌单的前20首歌曲&#xff0c;所以仅支持下载前20首歌曲&#xff08;非VIP音乐&…...

yarn包管理器在添加、更新、删除模块时,在项目中是如何体现的

技术很久不用&#xff0c;就变得生疏起来。对npm深受其害&#xff0c;决定对yarn再整理一遍。 yarn包管理器 介绍安装yarn帮助信息最常用命令 介绍 yarn官网&#xff1a;https://yarn.bootcss.com&#xff0c;学任何技术的最新知识&#xff0c;都可以通过其对应的网站了解。无…...

React实现Intro效果(基础简单)

下载&#xff1a;利用Intro.js实现简单的新手引导 npm install intro.js --save yarn add intro.js 第一步&#xff1a;在我们需要引导的页面引入 import introJs from intro.js; import intro.js/introjs.css; //css是下载成功后就有的 第二步&#xff1a;在组件页面 c…...

HBuilderx发布苹果的包需要注意什么

在HBuilderX中发布苹果的包&#xff0c;需要注意以下几点&#xff1a; 开发者账号注册&#xff1a;在发布应用到App Store之前&#xff0c;需要先注册一个苹果开发者账号。注册过程较为繁琐&#xff0c;需要提供个人信息并支付年费。应用标识和证书&#xff1a;在发布iOS应用之…...

智慧医疗能源事业线深度画像分析(上)

引言 医疗行业作为现代社会的关键基础设施,其能源消耗与环境影响正日益受到关注。随着全球"双碳"目标的推进和可持续发展理念的深入,智慧医疗能源事业线应运而生,致力于通过创新技术与管理方案,重构医疗领域的能源使用模式。这一事业线融合了能源管理、可持续发…...

鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个生活电费的缴纳和查询小程序

一、项目初始化与配置 1. 创建项目 ohpm init harmony/utility-payment-app 2. 配置权限 // module.json5 {"requestPermissions": [{"name": "ohos.permission.INTERNET"},{"name": "ohos.permission.GET_NETWORK_INFO"…...

IT供电系统绝缘监测及故障定位解决方案

随着新能源的快速发展&#xff0c;光伏电站、储能系统及充电设备已广泛应用于现代能源网络。在光伏领域&#xff0c;IT供电系统凭借其持续供电性好、安全性高等优势成为光伏首选&#xff0c;但在长期运行中&#xff0c;例如老化、潮湿、隐裂、机械损伤等问题会影响光伏板绝缘层…...

【LeetCode】算法详解#6 ---除自身以外数组的乘积

1.题目介绍 给定一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&#xff0c;且在 O…...

SQL Server 触发器调用存储过程实现发送 HTTP 请求

文章目录 需求分析解决第 1 步:前置条件,启用 OLE 自动化方式 1:使用 SQL 实现启用 OLE 自动化方式 2:Sql Server 2005启动OLE自动化方式 3:Sql Server 2008启动OLE自动化第 2 步:创建存储过程第 3 步:创建触发器扩展 - 如何调试?第 1 步:登录 SQL Server 2008第 2 步…...

前端高频面试题2:浏览器/计算机网络

本专栏相关链接 前端高频面试题1&#xff1a;HTML/CSS 前端高频面试题2&#xff1a;浏览器/计算机网络 前端高频面试题3&#xff1a;JavaScript 1.什么是强缓存、协商缓存&#xff1f; 强缓存&#xff1a; 当浏览器请求资源时&#xff0c;首先检查本地缓存是否命中。如果命…...

算术操作符与类型转换:从基础到精通

目录 前言&#xff1a;从基础到实践——探索运算符与类型转换的奥秘 算术操作符超级详解 算术操作符&#xff1a;、-、*、/、% 赋值操作符&#xff1a;和复合赋值 单⽬操作符&#xff1a;、--、、- 前言&#xff1a;从基础到实践——探索运算符与类型转换的奥秘 在先前的文…...

在golang中如何将已安装的依赖降级处理,比如:将 go-ansible/v2@v2.2.0 更换为 go-ansible/@v1.1.7

在 Go 项目中降级 go-ansible 从 v2.2.0 到 v1.1.7 具体步骤&#xff1a; 第一步&#xff1a; 修改 go.mod 文件 // 原 v2 版本声明 require github.com/apenella/go-ansible/v2 v2.2.0 替换为&#xff1a; // 改为 v…...

TCP/IP 网络编程 | 服务端 客户端的封装

设计模式 文章目录 设计模式一、socket.h 接口&#xff08;interface&#xff09;二、socket.cpp 实现&#xff08;implementation&#xff09;三、server.cpp 使用封装&#xff08;main 函数&#xff09;四、client.cpp 使用封装&#xff08;main 函数&#xff09;五、退出方法…...

高效的后台管理系统——可进行二次开发

随着互联网技术的迅猛发展&#xff0c;企业的数字化管理变得愈加重要。后台管理系统作为数据存储与业务管理的核心&#xff0c;成为了现代企业不可或缺的一部分。今天我们要介绍的是一款名为 若依后台管理框架 的系统&#xff0c;它不仅支持跨平台应用&#xff0c;还能提供丰富…...