外观


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

外观模式,即Facade,是一个比较简单的模式。它的基本思想如下:

如果客户端要跟许多子系统打交道,那么客户端需要了解各个子系统的接口,比较麻烦。如果有一个统一的“中介”,让客户端只跟中介打交道,中介再去跟各个子系统打交道,对客户端来说就比较简单。所以Facade就相当于搞了一个中介。

我们以注册公司为例,假设注册公司需要三步:

  1. 向工商局申请公司营业执照;
  2. 在银行开设账户;
  3. 在税务局开设纳税号。

以下是三个系统的接口:

  1. // 工商注册:
  2. public class AdminOfIndustry {
  3. public Company register(String name) {
  4. ...
  5. }
  6. }
  7. // 银行开户:
  8. public class Bank {
  9. public String openAccount(String companyId) {
  10. ...
  11. }
  12. }
  13. // 纳税登记:
  14. public class Taxation {
  15. public String applyTaxCode(String companyId) {
  16. ...
  17. }
  18. }

如果子系统比较复杂,并且客户对流程也不熟悉,那就把这些流程全部委托给中介:

  1. public class Facade {
  2. public Company openCompany(String name) {
  3. Company c = this.admin.register(name);
  4. String bankAccount = this.bank.openAccount(c.getId());
  5. c.setBankAccount(bankAccount);
  6. String taxCode = this.taxation.applyTaxCode(c.getId());
  7. c.setTaxCode(taxCode);
  8. return c;
  9. }
  10. }

这样,客户端只跟Facade打交道,一次完成公司注册的所有繁琐流程:

  1. Company c = facade.openCompany("Facade Software Ltd.");

很多Web程序,内部有多个子系统提供服务,经常使用一个统一的Facade入口,例如一个RestApiController,使得外部用户调用的时候,只关心Facade提供的接口,不用管内部到底是哪个子系统处理的。

更复杂的Web程序,会有多个Web服务,这个时候,经常会使用一个统一的网关入口来自动转发到不同的Web服务,这种提供统一入口的网关就是Gateway,它本质上也是一个Facade,但可以附加一些用户认证、限流限速的额外服务。

练习

使用Facade模式实现一个注册公司的“中介”服务。

外观 - 图1下载练习:Facade模式练习 (推荐使用IDE练习插件快速下载)

小结

Facade模式是为了给客户端提供一个统一入口,并对外屏蔽内部子系统的调用细节。

读后有收获可以支付宝请作者喝咖啡:

外观 - 图2