欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

浅述PHP设计模式(6)

程序员文章站 2022-05-11 10:14:53
...

三、行为型模式——调用行为的传递问题。

我们在前面已讨论了12种设计模式。其中涉及的是:变与不变,对变化如何处理,以及调用关系,对不可调用,如何方便实现成为可以调用。剩下的11个模式,都是与行为传递有关的,即不是变与不变的问题,也不是调用关系问题。而是调用流程控制的问题。这是因为,我们使用了类,使用了SOLID原则,调用流程不再是if else 或 switch case。 处理好这样的流程控制,则更易于我们的代码的简化。

    Chain of Responsibility(责任链):为解除请求的发送者和接受者之间耦合,而使多个对象都有机会处理这个请求。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它。
    这样的做法,使得每一个类中的流程控制得以统一。并且,可以将这样的流程控制封装到对应的类中,而不是由调用者来处理,因而调用者不再需要了解这个链中有多少个程员,该按怎样的顺序调用。从而丢开了调用者与被调用者之间的紧密耦合。
    实现方式:被调用者作为具体代码,其类中增加$next属性,保存下一个有责任的类。被调的方法中,判断,有没有$next,如果有,则调用$next中的同名方法。
    总结:一次性同时调用多个类中的同一个方法。(类方法递归)
    选择模式:通过类的链,实现方法选择的组合。

    Command(命令):将一个请求封装为一个对象,从而使你可以用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可取消的操作。
    实现方式:把对于一个类的方法的调用,封装成命令对象。调用时,不直接调用目标类的方法,而是调用命令对象。通过命令对象决定调用哪一个目标类中的哪一个目标方法。由此,调用者只与统一的命令对象接触,而不需要了解目标类的方法与细节。这样,使得目录类是可变的,并且,调用是可以加以限制的,比如:延时,撤消等。
    抽象的命令对象,定义命令的统计接口。命令对象中的构造函数的参数是命令的接收者。实际的命令通过继承命令对象来实现,从而实现抽象命令对象中的execute方法,或cancel方法,保证命令执行或取消。
    总结:命令对象的目标类可变,并且,调用类永远使用不变的命令。不管被调用的类中的方法是否变化。
    选择模式:类中的方法。

    Interpreter(解释器):给定一个语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子。
    解释器的根本目的,是处理多样的复杂指令,并且这些指令原本是散落在程序不同地方的。通过解释器,我们只需要一套这样的程序就够了,从而把散落在到处的算法给集中了起来。
    也许你认为,程序既然可以集中,那么,散落在各处的指令直接使用对象的方法调用不是一样吗?你错了。这样做调用方式就反了。解释器仍是核心调用具体。而不是让具体调用核心。如果是具体调用核心,那解释器就不需要了。
    实现方式:可以通过不同的语法节点的解释器聚合实现目标解释器类。
    总结:解释器是对输入的预处理,也可以用作输出的后处理。使其转换为程序的可用形式。
    选择模式:数据选择。


    Iterator(迭代器):提供一种方法顺序访问一个聚合对象中各个元素,而又不需暴露该对象的内部表示。
    实现方式:使用一个迭代器类,实现getCurrentX与getNextX方法。其中X是指你要访问的具体的数据部分。
    最简单的,PHP已经为我们封装好了。那就是PHP对象中的属性,我们可以用FOREACH进行访问,同时,如果我们使用SPL的ARRAYACCESS接口,也就可以用顺序访问。顺序访问使程序变得简单统一,并且调用者不再需要了解更多的细节。
    现实中也有很多这样类似的需求。比如工资表的打印,实际上是把查询结果中的一组记录要按同一员工ID变成一行记录。这时,如果用简单的FRO循环是不行的,而使用Iterator(迭代器),这样的工作就被封装到了类的内部。
    总结:通过一个类实现对复杂结构中某一项的顺序访问,实现封装,同时简化调用方的操作。
    选择模式:数据选择。


    Mediator(中介者):用一个中介对象来封装一系列的对象交互。中介者使各对象不需要显式地相互引用,从而使器耦合松散,而且可以独立地改变它们之间的交互。
    实现方式:创建一个Mediator类,保证当A与B双方都不确定时,运行时确定的A能直接访问Mediator,运行时确定的B能直接访问Mediator,这样即实现了任一A与任一B的互相操作。
    所以,作为Mediator类,要能够允许用户确定或变更调用方A,或被调用方B。并且,Mediator类负责将调用方A的调用传递给被调用方B。
    总结:Mediator不需要你知道你要访问的目标类及其方法是哪一个。
    选择模式:类及其方法选择。


    Memento(备忘录):在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可以将该对象恢复到保存的状态。
    实现方式:另建一个类,类名可以实际状态的名称,比如:BookMark表示书签,记录用户将此书读到了哪一页。
    Memento(备忘录)的根本目标是实现可撤消的变更,通过Memento类将对应类的数据或操作变更记录下来。这样,如果需要撤消,可以在Memento回朔寻找,从而撤消到指定的操作之前。
    总结:记住对象的的某一状态,以便可以回朔。
    选择模式:状态选择


    Observer(观察者):定义对象间的一种一对多的依赖关系,以便当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并自动刷新。
    实现方式:把对多个对象的更新操作封装到Observer的一个方法中。在Observer的方法中,我们可以根据不同的情况,决定需要同时更新哪些类。这即是具体的算法了。由此可见,这里,核心与具体的关系是,核心是调用与被调用方的类,而具体的是数据更新算法,如何更新,更新哪些类,则是要根据不同的情况决定的。
    总结:给调用方提供单一方法,方便操作。
    选择模式:方法选择

 (待续)