23.1 我要投递信件 我们都写过纸质信件吧,比如给女朋友写情书什么的。写信的过程大家应该都还记得 ——先写信的内容,然后写信封,再把信放到信封中,封好,投递到信箱中进行邮递,这个过程还是比较简单的,虽然简单,但是这4个步骤都不可或缺!我们先把这个过程通过程序实现出来,如图23-1所示。
图23-1 写信过程类图
这一个过程还是比较简单的,我们看程序的实现,先看接口,如代码清单23-1所示。
代码清单23-1 写信过程接口
1 2 3 4 5 6 7 8 9 10 public interface ILetterProcess { public void writeContext (String context) ; public void fillEnvelope (String address) ; public void letterInotoEnvelope () ; public void sendLetter () ; }
在接口中定义了完成的一个写信过程,这个过程需要实现,其实现类如代码清单23-2所示。
代码清单23-2 写信过程的实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 public class LetterProcessImpl implements ILetterProcess { 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("邮递信件..." ); } }
在这种环境下,最累的是写信人,为了发送一封信要有4个步骤,而且这4个步骤还不能颠倒,我们先看看这个过程如何通过程序表现出来,有人开始用这个过程写信了,如代码清单23-3所示。
代码清单23-3 场景类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 public class Client { public static void main (String[] args) { ILetterProcess 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(); } }
运行结果如下所示:
1 2 3 4 填写信的内容...Hello,It's me,do you know who I am? I'm your old lover.I'd like to... 填写收件人地址及姓名...Happy Road No.666,God Province,Heaven 把信放到信封中... 邮递信件...
我们回过头来看看这个过程,它与高内聚的要求相差甚远,更不要说迪米特法则、接口隔离原则了。你想想,你要知道这4个步骤,而且还要知道它们的顺序,一旦出错,信就不可能邮寄出去,这在面向对象的编程中是极度地不适合,它根本就没有完成一个类所具有的单一职责。
还有,如果信件多了就非常麻烦,每封信都要这样运转一遍,非得累死,更别说要发个广告信了,那怎么办呢?还好,现在邮局开发了一个新业务,你只要把信件的必要信息告诉我,我给你发,我来完成这4个过程,只要把信件交给我就成了,其他就不要管了。非常好的方案!我们来看类图,如图23-2所示。
图23-2 增加现代化邮局的类图
这还是比较简单的类图,增加了一个ModenPostOffice类,负责对一个比较复杂的信件处理过程的封装,然后高层模块只要和它有交互就成了,如代码清单23-4所示。
代码清单23-4 现代化邮局
1 2 3 4 5 6 7 8 9 10 11 12 13 14 public class ModenPostOffice { private ILetterProcess letterProcess = new LetterProcessImpl (); public void sendLetter (String context,String address) { letterProcess.writeContext(context); letterProcess.fillEnvelope(address); letterProcess.letterInotoEnvelope(); letterProcess.sendLetter(); } }
这个类是什么意思呢,就是说现在有一个Hell Road PostOffice(地狱路邮局)提供了一种新型服务,客户只要把信的内容以及收信地址给他们,他们就会把信写好,封好,并发送出去。这种服务推出后大受欢迎,这多简单,客户减少了很多工作,谁不乐意呀。那我们看看客户是怎么调用的,如代码清单23-5所示。
代码清单23-5 场景类
1 2 3 4 5 6 7 8 9 10 11 12 13 public class Client { public static void main (String[] args) { 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); } }
运行结果是相同的。我们看看场景类是不是简化了很多,只要与ModenPostOffice交互就 成了,其他的什么都不用管,写信封啦、写地址啦……都不用关心,只要把需要的信息提交 过去就成了,邮局保证会按照我们指定的地址把指定的内容发送出去,这种方式不仅简单, 而且扩展性还非常好,比如一个非常时期,寄往God Province(上帝省)的邮件都必须进行 安全检查,那我们就很好处理了,如图23-3所示。
图23-3 扩展后的系统类图
增加了一个Police类,负责对信件进行检查,如代码清单23-6所示。
代码清单23-6 信件检查类
1 2 3 4 5 6 public class Police { public void checkLetter (ILetterProcess letterProcess) { System.out.println(letterProcess+" 信件已经检查过了..." ); } }
我们再来看一下封装类ModenPostOffice的变更,它封装了这部分的变化,如代码清单23-7所示。
代码清单23-7 扩展后的现代化邮局
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 public class ModenPostOffice { private ILetterProcess 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变量的声明以及一个方法的调用,那这个写信的过程就变成这样:先写信、写信封,然后警察开始检查,之后才把信放到信封,最后发送出去,那这个变更对客户来说是透明的,他根本就看不到有人在检查他的邮件,他也不用了解,反正现代化的邮件系统都帮他做了,这也是他乐意的地方。
场景类还是完全相同,但是运行结果稍有不同,如下所示:
1 2 3 4 5 6 填写信的内容...Hello,It's me,do you know who I am?I'm your old lover.I'd like to... 填写收件人地址及姓名...Happy Road No.666,God Province,Heaven com.cbf4life.common3.LetterProcessImpl@15ff48b 信件已经检查过了... 把信放到信封中... 邮递信件...
高层模块没有任何改动,但是信件却已经被检查过了。这正是我们设计所需要的模式, 不改变子系统对外暴露的接口、方法,只改变内部的处理逻辑,其他兄弟模块的调用产生了不同的结果,确实是一个非常棒的设计。这就是门面模式。