简单描述
外观模式(Facade pattern),为子系统中的一组接口提供一个统一的高层接口,使得子系统更容易使用。
外观模式又称为门面模式,它是一种对象结构型模式,遵循迪米特法则,又称最少知道原则。
模式动机
现代的软件系统都非常复杂,尽管我们已经想尽一切方法将其“分而治之”,把一个系统划分为好几个较小的子系统了,但是仍然可能会存在这样的问题:子系统内有非常多的类,客户端往往需要和许多对象打交道之后 才能完成想要完成的功能。
在我们的生活中医院就是这样的。一般的医院都会分为挂号、门诊、化验、收费、取药等。看病的病人要想治好自己的病(相当于一个客户端想要实现自己的功能)就要和医院的各个部门打交道。首先,病人需要挂号,然后门诊,如果医生要求化验的话,病人就要去化验,然后再回到门诊室,最后拿药,经过一系列复杂的过程后才能完成看病的过程。如下图所示:
解决方案:
解决这种不便的方式就是引入外观模式。如果我们在医院设立一个接待员的话,病人只负责和接待员接触,由接待员负责与医院的各个部门打交道,让接待员完全帮助我们实现“看病”各个操作,如下图所示:
模式结构图
外观模式的基本组成:
外观角色 Facade:模式的核心,被客户 Client 调用,知道各个子系统的概念。根据客户角色的需求定制功能组合。
子系统角色 SubSystem:实现子系统的功能。
客户角色 Client:调用 Facade 角色获取相应的功能。
优点
- 降低系统的复杂程度,对客户屏蔽子系统组件,减少了客户处理的对象数目并使得子系统使用起来更加容易。通过引入外观模式,客户代码将变得很简单,与之关联的对象也很少。
- 实现了子系统与客户之间的松耦合关系,这使得子系统的组件变化不会影响到调用它的客户类,只需要调整外观类即可。
- 只是提供了一个访问子系统的统一入口,并不影响用户直接使用子系统类。
缺点
- 只是提供了一个访问子系统的统一入口,并不影响用户直接使用子系统类。
- 在不引入抽象外观类的情况下,增加新的子系统可能需要修改外观类或客户端的源代码,违背了“开闭原则”。
使用场景
- 当要为一个复杂子系统提供一个简单接口时可以使用外观模式。该接口可以满足大多数用户的需求,而且用户也可以越过外观类直接访问子系统。
- 客户程序与多个子系统之间存在很大的依赖性。引入外观类将子系统与客户以及其他子系统解耦,可以提高子系统的独立性和可移植性。
- 在层次化结构中,可以使用外观模式定义系统中每一层的入口,层与层之间不直接产生联系,而通过外观类建立联系,降低层之间的耦合度。
实例
下面以电脑开机为例子,正常情况下,我们只需要按下开机键即可,其他操作,比如 BIOS、打开操作系统这些操作是不需要用户关心的。
<?php
// 外观模式
// 下面以电脑开关机为例子,正常情况下,我们只需要按下开机键即可,其他操作,比如 BIOS、打开操作系统这些操作是不需要用户关心的。
class OperatingSystem {
public function open() {
echo '打开操作系统 ';
}
public function shutdown() {
echo '关闭操作系统 ';
}
public function login() {
echo '登录操作系统 ';
}
}
class Bios {
// 硬件自检
public function hardware_check() {
echo '硬件自检 ';
}
// 启动操作系统
public function launch(OperatingSystem $os) {
echo '启动操作系统 ';
}
// 电源关闭
public function power_down() {
echo '电源关闭 ';
}
}
class Facade {
private $os;
private $bios;
public function __construct() {
$this->bios = new Bios;
$this->os = new OperatingSystem;
}
public function turn_on() {
$this->bios->hardware_check();
$this->bios->launch($this->os);
$this->os->open();
$this->os->login();
}
public function turn_off() {
$this->os->shutdown();
$this->bios->power_down();
}
}
// client
$facade = new Facade();
// computer on
$facade->turn_on();
echo "<br>";
// computer off
$facade->turn_off();
输出结果:
硬件自检 启动操作系统 打开操作系统 登录操作系统
关闭操作系统 电源关闭
来源:oschina
链接:https://my.oschina.net/u/4414570/blog/4300768