门面模式,也叫外观模式。不管是门面还是外观,都是我们对外的媒介,就好像我们的脸面一样。所以,这个模式最大的特点就是要表现的“好看”。怎么说呢?一堆复杂的对象调用,自己都看蒙了,特别是对老系统进行升级维护的时候。用门面来把老系统的功能调用封装起来,在外面看来就和新系统一样,这就是门面模式的用途啦!

Gof类图及解释

GoF定义:为子系统中的一组接口提供一个一致的界面,Facade模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。

GoF类图

代码实现

  1. class SubSystemOne
  2. {
  3. public function MethodOne()
  4. {
  5. echo '子系统方法一', PHP_EOL;
  6. }
  7. }
  8. class SubSystemTwo
  9. {
  10. public function MethodTwo()
  11. {
  12. echo '子系统方法二', PHP_EOL;
  13. }
  14. }
  15. class SubSystemThree
  16. {
  17. public function MethodThree()
  18. {
  19. echo '子系统方法三', PHP_EOL;
  20. }
  21. }
  22. class SubSystemFour
  23. {
  24. public function MethodFour()
  25. {
  26. echo '子系统方法四', PHP_EOL;
  27. }
  28. }

定义四个或者N多个子系统,这个没什么好说的吧,可以想象是很多子系统,而且他们之间并不一定和这四个子系统一样的相似,有可能是千差万别的。

  1. class Facade
  2. {
  3. private $subStytemOne;
  4. private $subStytemTwo;
  5. private $subStytemThree;
  6. private $subStytemFour;
  7. public function __construct()
  8. {
  9. $this->subSystemOne = new SubSystemOne();
  10. $this->subSystemTwo = new SubSystemTwo();
  11. $this->subSystemThree = new SubSystemThree();
  12. $this->subSystemFour = new SubSystemFour();
  13. }
  14. public function MethodA()
  15. {
  16. $this->subSystemOne->MethodOne();
  17. $this->subSystemTwo->MethodTwo();
  18. }
  19. public function MethodB()
  20. {
  21. $this->subSystemOne->MethodOne();
  22. $this->subSystemTwo->MethodTwo();
  23. $this->subSystemThree->MethodThree();
  24. $this->subSystemFour->MethodFour();
  25. }
  26. }

通过门面类将这些子系统包装起来,对外提供的只是门面新定义的方法。

  1. $facade = new Facade();
  2. $facade->MethodA();
  3. $facade->MethodB();

客户端的调用就非常简单了,我们不用知道具体调用了哪些子系统,只需要知道门面的这些方法干什么了就行啦!

  • 门面模式就是这么的简单,而且只要是真实的在项目中做过开发的朋友一定在不知不觉中就已经使用过这个模式了
  • 当你需要为一个复杂子系统提供一个简单的接口时,门面模式就非常适用。同时,如果客户程序与抽象类的实现部分之间存在着很大的依赖性时,也可以引入门面模式来进行解耦,提高子系统的独立性和可维护性。另外就是你需要构建一个层次结构的子系统时,门面可以充当每层子系统的入口点
  • Laravel中的门面系统相信使用过框架的人一定都用过,比如:Cache::put()。在Laravel中,门面的实现使用了一个魔术方法__callStatic()。然后让对象的方法可以实现直接使用静态方法来进行调用。是不是很神奇。有兴趣的朋友可以翻翻源码,就在/Illuminate/Support/Facades/Facade.php中。
  • 划重点:三层结构或者MVC也是门面模式的体现哦。上面说了,门面模式适合分层子系统的维护。而三层结构、MVC、MVP、MVVM这些货,本质上都是为了分层,而分层的目的,就是为了降低系统的复杂性。

光卖我们的手机可不行,向X米一样做高科技的家电企业才是我们最终的目标。不过那么多的家电产品我们可生产不过来,于是,我们决定做一个商城(Facade)让一些质量非常好的商家加入我们的阵营,将他们的产品(SubSystem)放到商城中一起卖。当然,这些商品可是经过我们慎重挑选的,绝对都是优品中的优品哦!!

完整代码:https://github.com/zhangyue0503/designpatterns-php/blob/master/19.facade/source/facade.php

实例

这回我们将短信的发送以发送的维度进行包装,将不同的短信和推送运营商的接口包装起来,在发送的时候只需要通过发送类就可以控制用不同的第三方服务进行短信或推送的发送啦,想想都很方便呢!

短信发送类图

完整源码:https://github.com/zhangyue0503/designpatterns-php/blob/master/19.facade/source/facade-push.php

  1. <?php
  2. class Send
  3. {
  4. private $aliYunService;
  5. private $jiGuangService;
  6. private $message;
  7. private $push;
  8. public function __construct()
  9. {
  10. $this->aliYunService = new AliYunService();
  11. $this->jiGuangService = new JiGuangService();
  12. $this->message = new MessageInfo();
  13. $this->push = new PushInfo();
  14. }
  15. public function PushAndSendAliYun()
  16. {
  17. $this->message->Send($this->aliYunService);
  18. $this->push->Push($this->aliYunService);
  19. }
  20. public function PushAndSendJiGuang()
  21. {
  22. $this->message->Send($this->jiGuangService);
  23. $this->push->Push($this->jiGuangService);
  24. }
  25. }
  26. class MessageInfo
  27. {
  28. public function Send($service)
  29. {
  30. $service->Send();
  31. }
  32. }
  33. class PushInfo
  34. {
  35. public function Push($service)
  36. {
  37. $service->Push();
  38. }
  39. }
  40. class AliYunService
  41. {
  42. public function Send()
  43. {
  44. echo '发送阿里云短信!', PHP_EOL;
  45. }
  46. public function Push()
  47. {
  48. echo '推送阿里云通知!', PHP_EOL;
  49. }
  50. }
  51. class JiGuangService
  52. {
  53. public function Send()
  54. {
  55. echo '发送极光短信!', PHP_EOL;
  56. }
  57. public function Push()
  58. {
  59. echo '推送极光通知!', PHP_EOL;
  60. }
  61. }
  62. $send = new Send();
  63. $send->PushAndSendAliYun();
  64. $send->PushAndSendJiGuang();

说明

  • 依然还是熟悉的配方熟悉的味道。在这里,可以想象我们的第三方服务类都是较老的接口,或者已经是很复杂的接口了。这时,使用外观模式一来是可以与新系统配合,二来也能降低复杂度
  • 但是要注意的,外观类本身可能成为复杂度的来源,不过幸好我们可以遵从单一职责的原则,一个外观类就做一件事就好啦

下期看点

外观模式其实并没有太多可说的,因为它确实非常简单而且无处不在。不要被Laravel花哨的Facade类所欺骗,它并没有那么的复杂。下回我们学习到的依然是一个非常简单而且在不知不觉中就会用到的模式:模板方法模式

关注公众号:【硬核项目经理】获取最新文章

添加微信/QQ好友:【xiaoyuezigonggong/149844827】免费得PHP、项目管理学习资料

知乎、公众号、抖音、头条搜索【硬核项目经理】

B站ID:482780532

PHP设计模式之门面模式的更多相关文章

  1. python 设计模式之门面模式

    facade:建筑物的表面 门面模式是一个软件工程设计模式,主要用于面向对象编程. 一个门面可以看作是为大段代码提供简单接口的对象,就像类库.   门面模式被归入建筑设计模式.门面模式隐藏系统内部的细 ...

  2. 设计模式_Facade_门面模式

    形象例子: 我有一个专业的Nikon相机,我就喜欢自己手动调光圈.快门,这样照出来的照片才专业,但MM可不懂这些,教了半天也不会.幸好相机有Facade设计模式,把相机调整到自动档,只要对准目标按快门 ...

  3. JavaScript设计模式(6)-门面模式

    门面模式 门面模式(Facade Pattern):他隐藏了系统的复杂性,并向客户端提供了一个可以访问系统的接口.这种类型的设计模式属于结构性模式.为子系统中的一组接口提供了一个统一的访问接口,这个接 ...

  4. python设计模式之门面模式

    一.结构型设计模式 门面模式与单例模式,工厂模式不同,它是一种结构型模式. 结构型模式描述如何将对象和类组合成更大的结构 结构型模式是一种能够简化设计工作的模式,它能找出更简单的方法来认识或表示实体之 ...

  5. java设计模式之门面模式以及在java中作用

    门面模式在Tomcat中有多处使用,在Request和Response对象封装,从ApplicationContext到ServletContext封装中都用到了这种设计模式. 一个系统可以有几个门面 ...

  6. 【php设计模式】门面模式

    门面模式又叫外观模式,用来隐藏系统的复杂性,并向客户端提供了一个客户端可以访问系统的接口.这种类型的设计模式属于结构型模式,它向现有的系统添加一个接口,来隐藏系统的复杂性. 这种模式涉及到一个单一的类 ...

  7. JAVA设计模式之门面模式

    在阎宏博士的<JAVA与模式>一书中开头是这样描述门面(Facade)模式的: 门面模式是对象的结构模式,外部与一个子系统的通信必须通过一个统一的门面对象进行.门面模式提供一个高层次的接口 ...

  8. JS设计模式——10.门面模式

    门面模式 这是一种组织性的模式,它可以用来修改类和对象的接口,使其更便于使用.它可以让程序员过得更轻松,使他们的代码变得更容易管理. 门面模式有两个作用: 简化类的接口 消除与使用她的客户代码之间的耦 ...

  9. JAVA设计模式之门面模式(外观模式)

    医院的例子 现代的软件系统都是比较复杂的,设计师处理复杂系统的一个常见方法便是将其“分而治之”,把一个系统划分为几个较小的子系统.如果把医院作为一个子系统,按照部门职能,这个系统可以划分为挂号.门诊. ...

  10. JavaScript设计模式-12.门面模式

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

随机推荐

  1. CYPEESS USB3.0程序解读之---SPI读写

    前面已经解读了GPIO以及同步FIFO操作,下面我们看一个SPI读写的例子,它是主程序命令从SPI中读写一些数据. SPI传输子程序看一下: 页地址,字节计数,缓冲区,读写标志 因为只能一页一页的读或 ...

  2. 使用账号密码来操作github? NO!

    目录 简介 背景介绍 创建令牌 使用令牌 缓存令牌 使用GCM 总结 简介 最近在更新github文件的时候,突然说不让更新了,让我很是困惑,原因是在2021年8月13号之后,github已经不让直接 ...

  3. KNeighborsClassifier()的参数无效错误

    KNeighborsClassifier()的参数无效错误 写在前面:如果对自己英语拼写很自信,可以不用往下看了,我就是拼写错误 错误提示 Invalid parameter n_neightbors ...

  4. 温故知新,微软官方推荐的Visual Studio源代码管理之Git Ignore清单,开启新项目必备宝书

    什么是Git Ignore清单 https://git-scm.com/docs/gitignore 简单来说,在Git进行源代码管理中,我们可以通过建立.gitignore来实现一个忽略的黑名单管理 ...

  5. Apache Druid 远程代码执行 CVE-2021-25646 漏洞复现

    Apache Druid 远程代码执行 CVE-2021-25646 漏洞复现 前言 Apache Druid 是用Java编写的面向列的开源分布式数据存储,旨在快速获取大量事件数据,并在数据之上提供 ...

  6. NOIP 模拟 $19\; \rm u$

    题解 \(by\;zj\varphi\) 二维差分的题目 维护两个标记,一个向下传,一个向右下传: 对于每次更新,我们可以直接更新 \((r,c)+s,(r+l,c)-s\) ; \((r,c+1)- ...

  7. docker运行gerrit(代码审查工具)

    gerrit是什么? Gerrit,一种免费.开放源代码的代码审查软件,使用网页界面. gerrit背景 Gerrit,一种免费.开放源代码的代码审查软件,使用网页界面.利用网页浏览器,同一个团队的软 ...

  8. C# 调用C++结构体

    参考网址:C#调用C/C++动态库,封装各种复杂结构体._liguo9860的专栏-CSDN博客 现在公司要做一个使用C#程序调用C++的一个DLL库,解析文件的功能.所以在网上找了一些资料.     ...

  9. C#协作试取消线程

    https://segmentfault.com/q/1010000017109927using System; using System.Collections.Generic; using Sys ...

  10. 线程间协作的两种方式:wait、notify、notifyAll和Condition

    转载自海子: 在前面我们将了很多关于同步的问题,然而在现实中,需要线程之间的协作.比如说最经典的生产者-消费者模型:当队列满时,生产者需要等待队列有空间才能继续往里面放入商品,而在等待的期间内,生产者 ...