设计模式⑦ :简单化

文章目录

  • 一、前言
  • 二、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 角色之间的通信。

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

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

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

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

相关文章

配置redis挂载

1. 暂停和删除redis 2.创建文件夹 /usr/local/software/redis/6379/conf/ /usr/local/software/redis/6379/data/ 把redis-conf文件上传到conf文件夹中 3.配置网络 docker network create --driver bridge --subnet172.18.12.0/16 --gateway172.18.1.1 wn_docker_net 4.运…

Go后端开发 -- 反射reflect 结构体标签

Go后端开发 – 反射reflect && 结构体标签 文章目录 Go后端开发 -- 反射reflect && 结构体标签一、反射reflect1.编程语言中反射的概念2.interface 和反射3.变量内置的pair结构4.reflect的基本功能TypeOf和ValueOf5.从relfect.Value中获取接口interface的信息6…

C++ 设计模式之 中介者模式

【声明】本题目来源于卡码网&#xff08;题目页面 (kamacoder.com)&#xff09; 【提示&#xff1a;如果不想看文字介绍&#xff0c;可以直接跳转到C编码部分】 【设计模式大纲】 【简介】 -- 什么是中介者模式 &#xff08;第16种模式&#xff09; 中介者模式&#xff08;Medi…

数据结构->链队列(c/c++)

#include<iostream> #include<algorithm> #include<vector> using namespace std; //#define MAXSIZE 5 typedef int date_t; typedef struct listqueue {//节点结构体date_t name_id;listqueue* next; }listqueue,*list; typedef struct queue {//队列结构体…

Python 面向对象绘图(Matplotlib篇-16)

Python 面向对象绘图(Matplotlib篇-16)         🍹博主 侯小啾 感谢您的支持与信赖。☀️ 🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ�…

TQ8WS-acid,Tide Quencher 8WS-酸,可用来研究荧光物质的激发态

您好&#xff0c;欢迎来到新研之家 文章关键词&#xff1a;Tide Quencher8WS acid&#xff0c;TQ8WS acid&#xff0c;Tide Quencher 8WS 酸 &#xff0c;TQ8WS 酸&#xff0c;Tide Quencher 8WS-酸&#xff0c;TQ8WS-酸 一、基本信息 产品简介&#xff1a;The fluorescence…

牛客.KY11二叉树遍历、 LeetCode104.二叉树的最大深度 ,110平衡二叉树

二叉树实操小练习~这里对二叉树的遍历要有一定的理解&#xff0c;如果还不熟悉的小伙伴可以看看我的这篇博客&#xff1a;数据结构——二叉树&#xff08;先序、中序、后序及层次四种遍历&#xff08;C语言版&#xff09;&#xff09;超详细~ (✧∇✧) Q_Q-CSDN博客 牛客.KY11二…

对比分析ChatGPT 和文心一言。

目录 ChatGPT 和文心一言哪个更好用&#xff1f; ChatGPT 文心一言 对人工智能的看法 ChatGPT 和文心一言哪个更好用&#xff1f; ChatGPT和文心一言是目前两种比较流行的大语言模型&#xff0c;他们的出现迎来了人工智能时代。那么&#xff0c;这两种语言哪种更加好用一些…

Mybatis源码总结

mybatis Mybatis是一个orm框架&#xff0c;帮助我们更好的在java中编写和管理SQL语句 主要的运行过程&#xff1a; 主配置文件&#xff0c;配置mapper文件的位置&#xff0c;以及数据源&#xff0c;缓存&#xff0c;插件等配置信息&#xff0c;项目运行起来后会解析该文件&am…

数据库Mysql学习day01课堂笔记

一、数据库概述及数据准备 1.什么是数据库?什么是数据库管理系统?什么是SQL?他们之间的关系是什么? 数据库: 英文单词DateBase,简称DB。按照一定格式存储数据的一些文件的组合。顾名思义,存储数据的仓库,实际上就是一堆文件,这些文件中存储了具有特定格式的数据。 数…

Jtti:Linux内核怎么通过inline hook实现隐藏进程

在Linux内核中通过inline hook来隐藏进程是一种比较敏感和高级的操作&#xff0c;需要深入理解Linux内核的运作原理和对内核的深度了解。请注意&#xff0c;修改内核行为可能会导致系统不稳定&#xff0c;并且这种操作可能违反系统的安全策略&#xff0c;因此在进行此类操作之前…

民族服装数据研究:市场销售规模将突破百亿元

民族服饰文化内涵丰富&#xff0c;包括制作原料、纺织工艺、印染工艺、刺绣工艺、图案纹样、色彩表现、饰品工艺、文化价值等因素。我国民族服装在经历了过去30年全面工业化、经济高速增长、服装定制人口激增、大量生产、大量服装定制制消费到全国中产阶级化&#xff0c;人们对…

前端开发 5: Vue.js 框架

在前端开发中&#xff0c;Vue.js 是一个流行且灵活的 JavaScript 框架&#xff0c;用于构建用户界面。它采用了组件化的开发方式&#xff0c;使得前端开发更加模块化和可维护。在本篇博客中&#xff0c;我将为你介绍 Vue.js 的基础知识和常用技巧&#xff0c;帮助你更好地掌握前…

文档翻译网站有哪些?这些工具高效翻译

文档翻译网站有哪些&#xff1f;随着全球化的加速&#xff0c;跨语言沟通变得越来越重要。然而&#xff0c;语言差异常常成为我们与世界各地人们交流的障碍。为了解决这个问题&#xff0c;文档翻译软件应运而生。今天&#xff0c;我们就来介绍一些受欢迎的文档翻译软件&#xf…

【Docker】安装nacos以及实现负载均衡

&#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Docker的相关操作吧 目录 &#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 前言 一.nacos单个部署 1.镜像拉取 …

4.C++类和对象

深拷贝和浅拷贝的简单理解&#xff1a;

SpringBoot多环境配置及日志记录器

Spring Boot多环境配置 Spring Boot的针对不同的环境创建不同的配置文件&#xff0c; 语法结构&#xff1a;application-{profile}.properties profile:代表的就是一套环境 需求 application-dev.yml 开发环境 端口8090 application-test.yml 测试环境 端口8091 applica…

广东省第三届职业技能大赛“网络安全项目”B模块--数字取证解析

广东省第三届职业技能大赛“网络安全项目”B模块任务书 PS: 关注鱼影安全第一部分 网络安全事件响应第二部分 数字取证调查任务 3: 网络数据包分析取证解析:第三部分 应用程序安全:需要环境可以私信博主~PS: 关注鱼影安全 模块 B 竞赛项目试题 本文件为:广东省第三届职业技…

如何用mysql或者zk分配​​机器id

大家好&#xff0c;我是三叔&#xff0c;新的一年很高兴又和大家见面了&#xff0c;祝各位读者龙年大吉。 在 MySQL 中&#xff0c;可以使用自增主键来为每个记录分配唯一的机器 ID。创建一个包含自增主键的表&#xff0c;每当插入新记录时&#xff0c;MySQL 会自动为其分配一…