以下内容出自:<<24种设计模式介绍与6大设计原则>>

    好,我们继续讲课。大家都是高智商的人,都写过纸质的信件吧,比如给女朋友写情书什么的,写信的过程大家都还记得吧,先写信的内容,然后写信封,然后把信放到信封中,封好,投递到信箱中进行邮递,这个过程还是比较简单的,虽然简单,这四个步骤都是要跑的呀,信多了还是麻烦,比如到了情人节,为了大海捞针,给十个女孩子发情书,都要这样跑一遍,你不要累死,更别说你要发个广告信啥的,一下子发1 千万封邮件,那不就完蛋了?那怎么办呢?还好,现在邮局开发了一个新业务,你只要把信件的必要信息高速我,我给你发,我来做这四个过程,你就不要管了,只要把信件交给我就成了。

我们的类图还是从最原始的状态开始:

  在这中环境下,最累的是写信的人,为了发送一封信出去要有四个步骤,而且这四个步骤还不能颠倒,你不可能没写信就把信放到信封吧,写信的人要知道这四个步骤,而且还要知道这四个步骤的顺序,恐怖吧,我们先看看这个过程如何表现出来的:
先看写信的过程接口,定义了写信的四个步骤:

package cn.mjorcen.facade_pattern.savice;

/**
* @author cbf4Life cbf4life@126.com I'm glad to share my knowledge with you
* all. 定义一个写信的过程
*/
public interface LetterProcess {
// 首先要写信的内容
public void writeContext(String context); // 其次写信封
public void fillEnvelope(String address); // 把信放到信封里
public void letterInotoEnvelope(); // 然后邮递
public void sendLetter();
}

写信过程的具体实现:

package cn.mjorcen.facade_pattern.savice.impl;

import cn.mjorcen.facade_pattern.savice.LetterProcess;

/**
* @author cbf4Life cbf4life@126.com I'm glad to share my knowledge with you
* all. 写信的具体实现了
*/
public class LetterProcessImpl implements LetterProcess {
// 写信
public void writeContext(String context) {
System.out.println("填写信的内容...." + context);
} // 在信封上填写必要的信息
public void fillEnvelope(String address) {
System.out.println("填写收件人地址及姓名...." + address);
} // 把信放到信封中,并封好
public void letterInotoEnvelope() {
System.out.println("把信放到信封中....");
} // 塞到邮箱中,邮递
public void sendLetter() {
System.out.println("邮递信件...");
}
}

然后就有人开始用这个过程写信了:

package cn.mjorcen.facade_pattern;

import cn.mjorcen.facade_pattern.savice.LetterProcess;
import cn.mjorcen.facade_pattern.savice.impl.LetterProcessImpl; /**
* @author cbf4Life cbf4life@126.com I'm glad to share my knowledge with you
* all. 我开始给朋友写信了
*/
public class Client {
public static void main(String[] args) {
// 创建一个处理信件的过程
LetterProcess letterProcess = new LetterProcessImpl();
// 开始写信
letterProcess
.writeContext("Hello,It's me,do you know who I am? I'm your old lover. I'd like to....");
// 开始写信封
letterProcess.fillEnvelope("Happy Road No. 666,God Province,Heaven");
// 把信放到信封里,并封装好
letterProcess.letterInotoEnvelope();
// 跑到邮局把信塞到邮箱,投递
letterProcess.sendLetter();
}
}

  那这个过程与高内聚的要求相差甚远,你想,你要知道这四个步骤,而且还要知道这四个步骤的顺序,一旦出错,信就不可能邮寄出去,那我们如何来改进呢?先看类图:

  这就是门面模式,还是比较简单的,Sub System 比较复杂,为了让调用者更方便的调用,就对Sub System进行了封装,增加了一个门面,Client 调用时,直接调用门面的方法就可以了,不用了解具体的实现方法以及相关的业务顺序,我们来看程序的改变,LetterProcess 接口和实现类都没有改变,只是增加了一个
ModenPostOffice 类,我们这个java 程序清单如下:

package cn.mjorcen.facade_pattern.savice.impl;

import cn.mjorcen.facade_pattern.savice.LetterProcess;

/**
* @author cbf4Life cbf4life@126.com I'm glad to share my knowledge with you
* all.
*/
public class ModenPostOffice {
private LetterProcess letterProcess = new LetterProcessImpl(); // 写信,封装,投递,一体化了
public void sendLetter(String context, String address) {
// 帮你写信
letterProcess.writeContext(context);
// 写好信封
letterProcess.fillEnvelope(address);
// 把信放到信封中
letterProcess.letterInotoEnvelope();
// 邮递信件
letterProcess.sendLetter();
}
}

  这个类是什么意思呢,就是说现在又一个叫Hell Road PostOffice(地狱路邮局)提供了一种新型的服务,客户只要把信的内容以及收信地址给他们,他们就会把信写好,封好,并发送出去,这种服务提出时大受欢迎呀,这简单呀,客户减少了很多工作,那我们看看客户是怎么调用的,

Client.java 的程序清单
如下:

package cn.mjorcen.facade_pattern;

import cn.mjorcen.facade_pattern.savice.impl.ModenPostOffice;

/**
* @author cbf4Life cbf4life@126.com I'm glad to share my knowledge with you
* all. 我开始给朋友写信了
*/
public class Client {
public static void main(String[] args) {
// 现代化的邮局,有这项服务,邮局名称叫Hell Road
ModenPostOffice hellRoadPostOffice = new ModenPostOffice();
// 你只要把信的内容和收信人地址给他,他会帮你完成一系列的工作;
String address = "Happy Road No. 666,God Province,Heaven"; // 定义一个地址
String context = "Hello,It's me,do you know who I am? I'm your old lover.I'd like to....";
hellRoadPostOffice.sendLetter(context, address);
}
}

  看到没,客户简单了很多,提供这种模式后,系统的扩展性也有了很大的提高,突然一个非常时期,寄往God Province(上帝省)的邮件都必须进行安全检查,那我们这个就很好处理了,看类图:

看这个红色的框,只增加了这一部分,其他部分在类图上都不需要改动,那我们来看源码:

package cn.mjorcen.facade_pattern.savice.impl;

import cn.mjorcen.facade_pattern.savice.LetterProcess;

/**
* @author cbf4Life cbf4life@126.com I'm glad to share my knowledge with you
* all.
*/
public class ModenPostOffice {
private LetterProcess letterProcess = new LetterProcessImpl();
private Police letterPolice = new Police(); // 写信,封装,投递,一体化了
public void sendLetter(String context, String address) {
// 帮你写信
letterProcess.writeContext(context);
// 写好信封
letterProcess.fillEnvelope(address);
// 警察要检查信件了
letterPolice.checkLetter(letterProcess);
// 把信放到信封中
letterProcess.letterInotoEnvelope();
// 邮递信件
letterProcess.sendLetter();
}
}

  只是增加了一个letterPolice 变量的声明以及一个方法的调用,那这个写信的过程就变成了这样:先写信,然后写信封,然后警察开始检查,然后才把信放到信封,然后发送出去,那这个变更对客户来说,是透明的,他根本就看不到有人在检查他的邮件,他也不用了解,反正现代化的邮件都帮他做了,这也是他乐意的地方。

  门面模式讲解完毕,这是一个很好的封装方法,一个子系统比较复杂的实话,比如算法或者业务比较复杂,就可以封装出一个或多个门面出来,项目的结构简单,而且扩展性非常好。还有,在一个较大项目中的时候,为了避免人员带来的风险,也可以使用这个模式,技术水平比较差的成员,尽量安排独立的模块(Sub System),然后把他写的程序封装到一个门面里,尽量让其他项目成员不用看到这些烂人的代码,看也看不懂,我也遇到过一个“高人”写的代码,private 方法、构造函数、常量基本都不用,你要一个public 方法,好,一个类里就一个public 方法,所有代码都在里面,然后你就看吧,一大坨的程序,看着能把人逼疯,使用门面模式后,对门面进行单元测试,约束项目成员的代码质量,对项目整体质量的提升也是一个比较好的帮助。

第 7 章 门面模式【Facade Pattern】的更多相关文章

  1. php门面模式(facade pattern)

    书上不全的代码,我自己补全的. <?php /* The facade pattern is used when we want to simplify the complexities of ...

  2. 设计模式系列之外观模式(Facade Pattern)——提供统一的入口

    说明:设计模式系列文章是读刘伟所著<设计模式的艺术之道(软件开发人员内功修炼之道)>一书的阅读笔记.个人感觉这本书讲的不错,有兴趣推荐读一读.详细内容也可以看看此书作者的博客https:/ ...

  3. 说说设计模式~门面模式(Facade)

    返回目录 门面模式(Facade)属于结构型模式的一种,它符合面向对象的封装原则,但又不符合开闭原则,呵呵,今天我们主要说它的优点,不谈缺点. 定义 门面模式,是指提供一个统一的接口去访问多个子系统的 ...

  4. 二十四种设计模式:外观模式(Facade Pattern)

    外观模式(Facade Pattern) 介绍为子系统中的一组接口提供一个一致的界面,Facade模式定义了一个高层接口,这个接口使得这一子系统更加容易使用.示例有一个Message实体类,某对象对它 ...

  5. 乐在其中设计模式(C#) - 外观模式(Facade Pattern)

    原文:乐在其中设计模式(C#) - 外观模式(Facade Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 外观模式(Facade Pattern) 作者:webabcd 介绍 ...

  6. 使用C# (.NET Core) 实现适配器模式 (Adapter Pattern) 和外观模式 (Facade Pattern)

    本文的概念内容来自深入浅出设计模式一书 现实世界中的适配器(模式) 我带着一个国标插头的笔记本电脑, 来到欧洲, 想插入到欧洲标准的墙壁插座里面, 就需要用中间这个电源适配器. 面向对象的适配器 你有 ...

  7. 外观模式 门面模式 Facade 结构型 设计模式(十三)

    外观模式(FACADE) 又称为门面模式   意图 为子系统中的一组接口提供一个一致的界面 Facade模式定义了一个高层接口,这一接口使得这一子系统更加易于使用. 意图解析 随着项目的持续发展,系统 ...

  8. JAVA设计模式——第 7 章 门面模式【Facade Pattern】(转)

    好,我们继续讲课.大家都是高智商的人,都写过纸质的信件吧,比如给女朋友写情书什么的,写信的过程大家都还记得吧,先写信的内容,然后写信封,然后把信放到信封中,封好,投递到信箱中进行邮递,这个过程还是比较 ...

  9. 设计模式(八): 从“小弟”中来类比"外观模式"(Facade Pattern)

    在此先容我拿“小弟”这个词来扯一下淡.什么是小弟呢,所谓小弟就是可以帮你做一些琐碎的事情,在此我们就拿“小弟”来类比“外观模式”.在上面一篇博文我们完整的介绍了“适配器模式”,接下来我们将要在这篇博客 ...

随机推荐

  1. iOS 推送,当接到推送消息时如何处理?

    接收到通知时有两种进入的方式:1.当app未运行时(BOOL)application:(UIApplication *)application didFinishLaunchingWithOption ...

  2. [转]Form Builder:app_field.clear_dependent_fields和APP_FIELD.set_dependent_field的用法

    转自:http://www.cnblogs.com/toowang/p/3668070.html 可以调用APP_FIELD.clear_dependent_fields和APP_FIELD.set_ ...

  3. eclipse 注释模板设置

    方法注释模板 /** * @title ${enclosing_method} * @description ${todo} * ${tags} ${return_type} * @Date ${da ...

  4. Unity3D 之UGUI 面板

    UGUI中使用面板,可以对一组控件分为一个组. 一个面板里面可以添加一些控件,就如同Html中的<div>一样.

  5. XML操作 之获取指定节点值

    根据节点名称快速查找 指定节点值 using (TextReader stringReader = new StringReader(clearPassResponse)) { XmlReaderSe ...

  6. SQL语句一点小心得

    在Sqlserver中 if 语句后面的语句加begin end 括起来 问题:执行速度问题,在存储过程中没有加begin end 执行速度很慢,加了begin end执行速度加快 ALTER PRO ...

  7. mysql分表,分区的区别和联系

    一,什么是mysql分表,分区 什么是分表,从表面意思上看呢,就是把一张表分成N多个小表,具体请看mysql分表的3种方法 什么是分区,分区呢就是把一张表的数据分成N多个区块,这些区块可以在同一个磁盘 ...

  8. UItextField常用方法

    - (void)viewDidLoad {     [super viewDidLoad];     // Do any additional setup after loading the view ...

  9. PAT_1007 素数对猜想

    今天想更的那道题现在还没A出来.先把下午做的一道题更新了吧.快零点了.无奈啊. 问题描述: 让我们定义 dn 为:dn = pn+1 - pn,其中 pi 是第i个素数.显然有 d1=1 且对于n&g ...

  10. 2015 Multi-University Training Contest 1 题解&&总结

    ---------- HDU 5288 OO’s Sequence 题意 给定一个数列(长度<$10^5$),求有多少区间[l,r],且区间内有多少数,满足区间内其它数不是他的约数. 数的范围$ ...