设计模式⑦ :简单化

文章目录

  • 一、前言
  • 二、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.hqwc.cn/news/415194.html

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

相关文章

为 OpenCV 编写文档(二)

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

亚马逊卖食品有什么具体要求?亚马逊卖食品好做吗?—站斧浏览器

亚马逊卖食品有什么具体要求&#xff1f; 首先&#xff0c;亚马逊要求卖家提供食品的详细信息&#xff0c;包括产品描述、成分表、营养信息和包装规格等。这些信息对于消费者来说至关重要&#xff0c;它们可以帮助消费者了解产品的特点和质量&#xff0c;并做出明智的购买决策…

51单片机原理及应用——张毅刚版本代码全集可复制

从左到右的流水灯的制作(重点) #include <reg51.h> #include <intrins.h> //移位函数的头文件 unsigned int j; void Delay(unsigned int i) // 延时函数 {while(i--) // 注意这里是i--{for(j 0;j < 120;j); // 注意这里是120} }void main() {P1 0xFE;while(1…

INS-06003错误处理

在麒麟V10操作系统上安装Oracle RAC 19C&#xff0c;安装GI的建立互信步骤中&#xff0c;遇到INS-06003错误&#xff1a; [INS-06003] Failed to setup password SSH connectivity with following node(s) 查看详细信息&#xff1a; PRVG-11001: PRCZ-2136: PRCZ-2006: 此时在操…

leetcode82. 删除排序链表中的重复元素 II

文章目录 题目思路1复杂度Code2 思路2复杂度2Code2 题目 给定一个已排序的链表的头 head &#xff0c; 删除原始链表中所有重复数字的节点&#xff0c;只留下不同的数字 。返回 已排序的链表 。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,3,4,4,5] 输出&#xff1a;…

SpringBoot Redis入门(四)——Redis单机、哨兵、集群模式

单机模式&#xff1a;单台缓存服务器&#xff0c;开发、测试环境下使用&#xff1b;哨兵模式&#xff1a;主-从模式&#xff0c;提高缓存服务器的高可用和安全性。所有缓存的数据在每个节点上都一致。每个节点添加监听器&#xff0c;不断监听节点可用状态&#xff0c;一旦主节点…

CTF CRYPTO 密码学-1

题目名称&#xff1a;enc 题目描述&#xff1a; 压缩包中含两个文件&#xff1a;一个秘钥d.dec&#xff0c;一个密文flag.enc 解题过程&#xff1a; Step1&#xff1a;这题是一个解密他题目&#xff0c;尝试openssl去ras解密 工具简介 在Kali Linux系统中&#xff0c;openss…

深入理解 Kubernetes Ingress:路由流量、负载均衡和安全性配置

Kubernetes Ingress 是 Kubernetes 集群中外部流量管理的重要组件。它为用户提供了一种直观而强大的方式&#xff0c;通过定义规则和配置&#xff0c;来控制外部流量的路由和访问。 1. 什么是 Ingress&#xff1f; 在 Kubernetes 中&#xff0c;Ingress 是一种 API 资源&#…

C语言--质数算法和最大公约数算法

文章目录 1.在C语言中&#xff0c;判断质数的常见算法有以下几种&#xff1a;1.1.试除法&#xff08;暴力算法&#xff09;&#xff1a;1.2.优化试除法&#xff1a;1.3.埃拉托色尼筛法&#xff1a;1.4.米勒-拉宾素性检验&#xff1a;1.5.线性筛法&#xff1a;1.6.费马小定理&am…

LLM:Scaling Laws for Neural Language Models (中)

核心结论 1&#xff1a;LLM模型的性能主要与计算量C&#xff0c;模型参数量N和数据大小D三者相关&#xff0c;而与模型的具体结构 (层数/深度/宽度) 基本无关。三者满足: C ≈ 6ND 2. 为了提升模型性能&#xff0c;模型参数量N和数据大小D需要同步放大&#xff0c;但模型和数…

赋能客户不停歇,卓翼飞思职业院校无人智能装备师资培训圆满落幕

1月10日-15日&#xff0c;卓翼飞思在北京研发中心成功举办职业院校无人智能装备高级师资培训。来自湖南汽车工程职业学院&#xff08;以下简称&#xff1a;湖南汽车职院&#xff09;的10名师生&技术人员参与此次培训&#xff0c;针对无人智能集群应用开发进行系统性的培训及…

重生奇迹MU快速升级

1、前期&#xff0c;都是主线任务&#xff0c;我们必须要跟着主线任务走&#xff0c;前面的话升级一次需要的经验很少的&#xff0c;一天下来可以升级100级是轻轻松松的&#xff0c;主线任务是比较多的&#xff0c;我们跟着任务一直做差不多可以到150级&#xff0c;接下来就是后…