设计模式(十)外观模式

栏目: 后端 · 发布时间: 5年前

内容简介:外观模式是一种使用频率非常高的结构型设计模式, 它通过引入一个外观角色来简化客户端和子系统之间的交互, 为复杂的子系统提供统一的入口, 降低子系统与客户端的耦合度, 且客户端调用非常方便.在软件开发中, 有时候为了完成一项较为复杂的功能, 一个客户类需要和多个业务类交互, 而这些需要交互的业务类经常作为一个整体出现, 由于涉及到的类比较多, 导致使用时候代码较为复杂, 此时特别需要一个类似服务员一样的角色, 由他来负责和多个业务类进行交互, 而客户端只需要与该类交互. 外观模式中, 引入一个新的外观类(Fa

外观模式是一种使用频率非常高的结构型设计模式, 它通过引入一个外观角色来简化客户端和子系统之间的交互, 为复杂的子系统提供统一的入口, 降低子系统与客户端的耦合度, 且客户端调用非常方便.

1. 外观模式概述

在软件开发中, 有时候为了完成一项较为复杂的功能, 一个客户类需要和多个业务类交互, 而这些需要交互的业务类经常作为一个整体出现, 由于涉及到的类比较多, 导致使用时候代码较为复杂, 此时特别需要一个类似服务员一样的角色, 由他来负责和多个业务类进行交互, 而客户端只需要与该类交互. 外观模式中, 引入一个新的外观类(Facade)来实现该功能, 外观类充当软件系统中的"服务员", 它为多个业务类的调用提供一个统一的入口, 简化了类与类之间的交互. 在外观模式中, 那些需要交互的业务类被称为子系统(SubsSystem). 如果没有外观类,那么每个客户类需要和多个子系统之间进行复杂的交互,系统的耦合度将很大;而引入外观类之后,客户类只需要直接与外观类交互,客户类与子系统之间原有的复杂引用关系由外观类来实现,从而降低了系统的耦合度。

在外观模式中, 一个子系统的外部与其内部的通信通过一个统一的外观类进行, 外观类将客户类和子系统的内部的复杂性分隔开, 使得客户类只需与外观角色打交道, 而不需要与子系统内部的很多对象打交道.

外观模式: 为子系统中的一组接口提供统一的入口. 外观模式定义一个高层接口, 这个接口使得这一子系统更加容易被使用.

外观模式又称为门面模式, 它是一种对象结构型模式. 外观模式是迪米特法则的一种具体实现, 通过引入一个新的外观角色可以降低原有系统的复杂度, 同时降低客户类与子系统的耦合度.

2. 外观模式的实现

例一:

给自己举个最熟悉的例子: MJRefresh, 其中 UIScrollView+MJRefresh 就是一个外观类. 类抛出的方法和属性如下所示, 还是蛮简单的, 使用时候, 我们只需要和这个外观类进行交互, 将自己需要的header或者footer注入到外观类中, 到底什么时候用到这些对象什么时候会调用待header的相关方法, 就不是我们关系的事情了, 这些复杂的逻辑都被放在外观类中进行处理,.

这样, 使用框架的 程序员 们和封装框架的人之间通过MJRefresh这个扩展联系起来, 我们不需要关心MJRefresh内部实现多么复杂, 也不必关心它的内部逻辑, 可以通过外观类简单的使用MJRefresh. 这就是外观模式.

class MJRefreshHeader, MJRefreshFooter;

@interface UIScrollView (MJRefresh)
/** 下拉刷新控件 */
@property (strong, nonatomic) MJRefreshHeader *mj_header;
@property (strong, nonatomic) MJRefreshHeader *header MJRefreshDeprecated("使用mj_header");
/** 上拉刷新控件 */
@property (strong, nonatomic) MJRefreshFooter *mj_footer;
@property (strong, nonatomic) MJRefreshFooter *footer MJRefreshDeprecated("使用mj_footer");

#pragma mark - other
- (NSInteger)mj_totalDataCount;
@property (copy, nonatomic) void (^mj_reloadDataBlock)(NSInteger totalDataCount);
@end

复制代码

例二:

比如一个软件的文件加密模块, 该软件可以将文件加密然后把存储到一个新文件中, 具体流程分为: 读取文件,加密, 保存加密文件. 为了让代码独立重用, 让设计更加符合单一职责原则, 将这三个操作的业务代码封装到三个不同的类中.

使用外观类组合加密操作, 如下图, 其中EncryptFacade是一个外观类, 加密操作分为三个操作, 但是我们在使用加密时候不必关心加密的过程, 只需要调用 FileEncrypt方法即可完成读取/加密/写入新文件.

3. 抽象外观类

如上例二途中所示, 标准的外观模式中, 如果要更换外观类的子系统, 则必须修改外观类的客户端源代码, 这将违背开闭原则. 因此可以引进抽象外观类来对系统进行改进, 在一定程度上可以解决该问题. 引入抽象外观类之后, 客户端可以针对抽象外观类进行编程, 对于新的业务需求可以在不修改原有外观类的情况下修改具体的实现类. 这样可以适配不同的加密算法.

4. 外观模式效果与适用场景

外观模式是一种使用频率非常高的设计模式,它通过引入一个外观角色来简化客户端与子系统之间的交互,为复杂的子系统调用提供一个统一的入口,使子系统与客户端的耦合度降低,且客户端调用非常方便。外观模式并不给系统增加任何新功能,它仅仅是简化调用接口。在几乎所有的软件中都能够找到外观模式的应用,如绝大多数B/S系统都有一个首页或者导航页面,大部分C/S系统都提供了菜单或者 工具 栏,在这里,首页和导航页面就是B/S系统的外观角色,而菜单和工具栏就是C/S系统的外观角色,通过它们用户可以快速访问子系统,降低了系统的复杂程度。所有涉及到与多个业务对象交互的场景都可以考虑使用外观模式进行重构。

4.1 模式优点

外观模式的主要优点如下:

  • (1) 它对客户端屏蔽了子系统组件,减少了客户端所需处理的对象数目,并使得子系统使用起来更加容易。通过引入外观模式,客户端代码将变得很简单,与之关联的对象也很少。

  • (2) 它实现了子系统与客户端之间的松耦合关系,这使得子系统的变化不会影响到调用它的客户端,只需要调整外观类即可。

  • (3) 一个子系统的修改对其他子系统没有任何影响,而且子系统内部变化也不会影响到外观对象。

4.2 模式缺点

外观模式的主要缺点如下:

  • (1) 不能很好地限制客户端直接使用子系统类,如果对客户端访问子系统类做太多的限制则减少了可变性和灵活 性。
  • (2) 如果设计不当,增加新的子系统可能需要修改外观类的源代码,违背了开闭原则。

4.3 模式适用场景

在以下情况下可以考虑使用外观模式:

  • (1) 当要为访问一系列复杂的子系统提供一个简单入口时可以使用外观模式。

  • (2) 客户端程序与多个子系统之间存在很大的依赖性。引入外观类可以将子系统与客户端解耦,从而提高子系统的独立性和可移植性。

  • (3) 在层次化结构中,可以使用外观模式定义系统中每一层的入口,层与层之间不直接产生联系,而通过外观类建立联系,降低层之间的耦合度。

角色:

1,外观(Facade)角色 :客户端可以调用这个角色的方法。此角色知晓相关子系统的功能和责任。在正常情况下,本角色会将所有从客户端发来的请求委派到相应的子系统去。

2,子系统(SubSystem)角色 :可以同时有一个或者多个子系统。每个子系统都不是一个单独的类,而是一个类的集合。每个子系统都可以被客户端直接调用,或者被外观角色调用。子系统并不知道外观角色的存在,对于子系统而言,外观角色仅仅是另外一个客户端而已。

设计模式(十)外观模式

示例:

1,子系统角色,由若干个类组成

public class SubClass1 {    
    public void method1(){
        System.out.println("这是子系统类1中的方法1");
    }    
    public void method2(){
        System.out.println("这是子系统类1中的方法2");
    }
}
public class SubClass2 {    
    public void method1(){
        System.out.println("这是子系统类2中的方法1");
    }    
    public void method2(){
        System.out.println("这是子系统类2中的方法2");
    }
}
public class SubClass3 {   
    public void method1(){
        System.out.println("这是子系统类3中的方法1");
    }    
    public void method2(){
        System.out.println("这是子系统类3中的方法2");
    }
}
复制代码

2,外观角色类

public class FacadeClass {    
    public void FacadeMethod(){
        SubClass1 s1 = new SubClass1();
        s1.method1();
        SubClass2 s2 = new SubClass2();
        s2.method1();
        SubClass3 s3 = new SubClass3();
        s3.method1();
    }
}
复制代码

3,客户端测试方法

public class ClientClass {
    public static void main(String[] args) {
        FacadeClass fc = new FacadeClass();
        fc.FacadeMethod();
    }
}
复制代码

Facade类其实相当于子系统中SubClass类的外观界面,有了这个Facade类,那么客户端就不需要亲自调用子系统中的那些具体实现的子类了,也不需要知道系统内部的实现细节,甚至都不需要知道这些子类的存在,客户端只需要跟Facade类交互就好了,从而更好地实现了客户端和子系统中具体类的解耦,让客户端更容易地使用系统。

同时,这样定义一个Facade类可以有效地屏蔽内部的细节,免得客户端去调用Module类时,发现一些不需要它知道的方法。如上代码,我的所有子类中的方法二都是方法一调用的方法,是配合方法一的,他们不需要被客户端调用,而且具有一定的保密性,这样使用外观模式时就可以不被客户端知道。

note:你总不能因为喝水的杯子摔坏了就再也不喝水了吧


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

Java程序设计

Java程序设计

宋中山 严千钧 等编 / 清华大学出版社 / 2005-8 / 27.00元

本书全面、系统地介绍了Java语言的基本概念、基本语法和编程方法。主要内容包括:Java语言概述、数据类型与运算符、流程控制语句、类与对象、继承与多态、异常处理、工具类和算法、Applet小应用程序、图形用户界面、输入和输出、Java多线程以及Java高级编程。每章后面附有习题,读者可参考使用。 本书内容丰富,结构合理,语言简洁,深入浅出,通俗易懂。基础知识与程序实例相结合,示例典型......一起来看看 《Java程序设计》 这本书的介绍吧!

JSON 在线解析
JSON 在线解析

在线 JSON 格式化工具

Base64 编码/解码
Base64 编码/解码

Base64 编码/解码

XML 在线格式化
XML 在线格式化

在线 XML 格式化压缩工具