设计模式-Facade

结构型模式

组成

门面模式/外观模式(Facade Pattern)通过定义了一个高层次的接口,为子系统中的一组接口提供一个一致的界面,使得这个子系统更加容易使用。

屏蔽子系统一些复杂的内部行为

门面模式最重要的是创建一个门面类,该门面类对外代表了一个子系统,如下图所示:

上图中的参与者有:

  • Facade:把客户的请求转发给子系统中的某个对象;
  • 子系统:提供功能的子系统,没有 Facade 的任何信息,不持有 Facade 的引用。

在上图中,如果没有 Facade 类,那么 Client 在与子系统交互时,需要依赖子系统中的所有对象,增加了系统的复杂程度。通过引入一个 Facade 对象,它为子系统提供一个单一而简单的界面,这样 Client 只需要依赖 Facade 对象即可。

应用场景

当存在以下情况时,推荐使用 门面模式:

  • 当我们要为一个复杂子系统提供一个简单的接口时,Facade 对象提供一些基本的方法,以满足常规的需求;如果需要使用更复杂功能的场景,也可以直接绕过 Facade 使用子系统中的对象。
  • 当我们需要构建一个分层的系统时,使用 Facade 模式定义每一层系统的入口;如果同一层的子系统之间相互依赖,可以让它们通过 Facade 进行通信,简化它们的依赖关系。

微服务架构中的一个服务,可以看作是一个子系统,它对外提供的的SDK可以看作这个服务的 Facade 对象。

示例代码

tomcat 给 Request 类提供一个 Facade 类:RequestFacade,用来屏蔽 Request 的复杂操作,也避免用户受到非 servlet 标准方法的干扰。

RequestFacade 与 Request 的 UML 如下图所示:

当我们在 Spring 容器或者接口中调用 HttpServletRequest 时,实际调用的是 RequestFacade 对象。当我们调用它的方法时,都会转发给对应的 Request request 对象。

示例代码如下:

java 复制代码
/**
 * The wrapped request.
 */
protected Request request = null;

public RequestFacade(Request request) {

    this.request = request;
}

@Override
public Object getAttribute(String name) {

    if (request == null) {
        throw new IllegalStateException(
            sm.getString("requestFacade.nullRequest"));
    }

    return request.getAttribute(name);
}

这里需要注意的是,RequestFacade 和 Request 都实现了 HttpServletRequest 接口,并且 RequestFacade 持有 Request 对象,也就是说 Facade 继承了要屏蔽的对象。

其实一般是不用这么做的,这里之所以这么做,是因为服务中后续对 Request 的操作都是在操作 HttpServletRequest 对象,所以这里 RequestFacade 才要实现 HttpServletRequest,否则无法继续向下继续传递。

优点和缺点

门面模式的优点:

  • 因为减少了客户端需要依赖的子系统对象,使得子系统使用起来更加方便;
  • 实现了子系统与客户之间的松耦合关系,使得子系统内部的变化不会影响到客户端;
  • Facade 模式有助于建立层次结构系统,有助于对象之间的依赖关系分层;
  • Facade 不限制客户端使用子系统中的对象,具有很大的选择灵活性;

小结

Facade 模式可以为一个子系统提供统一的简单界面,降低客户端与子系统之间的耦合。

Facade 和 Proxy 看上去有些相似,但是两者是有很大区别的:

  • Proxy 是代理某个对象,给被代理对象增加一些非逻辑的功能
  • Facade 是作为一个子系统的"门面",不会增加功能,而是屏蔽复杂性,请求是被直接转发到子系统的
相关推荐
大鱼七成饱14 分钟前
💥 从崩溃到稳定:我踩过的 Rust Tokio 线程池坑(含代码示例)
后端
喵个咪31 分钟前
开箱即用的GO后台管理系统 Kratos Admin - 站内信
后端·微服务·go
韩立学长1 小时前
基于Springboot的旧物公益捐赠管理系统3726v22v(程序、源码、数据库、调试部署方案及开发环境)系统界面展示及获取方式置于文档末尾,可供参考。
数据库·spring boot·后端
Dyan_csdn1 小时前
springboot系统设计选题3
java·spring boot·后端
Yeats_Liao2 小时前
时序数据库系列(二):InfluxDB安装配置从零搭建
数据库·后端·时序数据库
Yeats_Liao2 小时前
时序数据库系列(一):InfluxDB入门指南核心概念详解
数据库·后端·时序数据库·db
蓝-萧2 小时前
springboot系列--自动配置原理
java·后端
bobogift3 小时前
【玩转全栈】----Django基本配置和介绍
java·后端
倚栏听风雨3 小时前
Async-Profiler 框架简介
后端
qianbailiulimeng3 小时前
2019阿里java面试题(一)
java·后端