PHP设计模式—访问者模式

定义:

访问者模式(Visitor):表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。

结构:

  • Visitor:抽象访问者,为该对象结构中ConcreteElement的每一个类声明一个Visit操作。
  • ConcreteVisitor:具体访问者,实现每个由Visitor声明的操作。每个操作实现算法的一部分,而该算法片段乃是对应于结构中对象的类。
  • Element:抽象类,定义一个Accept操作,它以一个访问者为参数。
  • ConcreteElement:具体元素,实现Accept操作。
  • ObjectStructure:对象结构,能枚举它的元素,可以提供一个高层的接口以允许访问者访问它的元素。
  • Client:客户端代码。

代码实例:

/**
 * 抽象访问者
 * Class VisitorState
 */
abstract class VisitorState
{
    public $state_name;

    /**
     * 得到男人的结论或反应
     * @param ConcreteElementMan $concreteElementMan
     * @return mixed
     */
    abstract public function getManConclusion(ConcreteElementMan $concreteElementMan);

    /**
     * 得到女人的结论或反应
     * @param ConcreteElementWoman $concreteElementWoman
     * @return mixed
     */
    abstract public function getWomanConclusion(ConcreteElementWoman $concreteElementWoman);
}

/**
 * 抽象类,定义一个Accept操作
 * Class ElementPerson
 */
abstract class ElementPerson
{
    public $type_name;

    /**
     * 接受
     * @param VisitorState $visitorState
     * @return mixed
     */
    abstract public function accept(VisitorState $visitorState);
}

/**
 * 具体访问者,成功状态
 * Class ConcreteVisitorSuccess
 */
class ConcreteVisitorSuccess extends VisitorState
{
    public function __construct()
    {
        $this->state_name = '成功';
    }

    public function getManConclusion(ConcreteElementMan $concreteElementMan)
    {
        // TODO: Implement getManConclusion() method.
        echo "{$concreteElementMan->type_name}:{$this->state_name}时,背后多半有一个伟大的女人。<br/>";
    }

    public function getWomanConclusion(ConcreteElementWoman $concreteElementWoman)
    {
        // TODO: Implement getWomanConclusion() method.
        echo "{$concreteElementWoman->type_name}:{$this->state_name}时,背后大多有一个不成功的男人。<br/>";
    }
}

/**
 * 具体访问者,失败状态
 * Class ConcreteVisitorFailing
 */
class ConcreteVisitorFailing extends VisitorState
{
    public function __construct()
    {
        $this->state_name = '失败';
    }

    public function getManConclusion(ConcreteElementMan $concreteElementMan)
    {
        // TODO: Implement getManConclusion() method.
        echo "{$concreteElementMan->type_name}:{$this->state_name}时,闷头喝酒,谁也不用劝。<br/>";
    }

    public function getWomanConclusion(ConcreteElementWoman $concreteElementWoman)
    {
        // TODO: Implement getWomanConclusion() method.
        echo "{$concreteElementWoman->type_name}:{$this->state_name}时,眼泪汪汪,谁也劝不了。<br/>";
    }
}

/**
 * 具体元素,男人
 * Class ConcreteElementMan
 */
class ConcreteElementMan extends ElementPerson
{
    public function __construct()
    {
        $this->type_name = '男人';
    }

    public function accept(VisitorState $visitorState)
    {
        // TODO: Implement accept() method.
        $visitorState->getManConclusion($this);
    }
}

/**
 * 具体元素,女人
 * Class ConcreteElementWoman
 */
class ConcreteElementWoman extends ElementPerson
{
    public function __construct()
    {
        $this->type_name = '女人';
    }

    public function accept(VisitorState $visitorState)
    {
        // TODO: Implement accept() method.
        $visitorState->getWomanConclusion($this);
    }
}

/**
 * 对象结构
 * Class ObjectStructure
 */
class ObjectStructure
{
    public $elements = [];

    /**
     * 添加
     * @param ElementPerson $elementPerson
     */
    public function add(ElementPerson $elementPerson)
    {
        $this->elements[] = $elementPerson;
    }

    /**
     * 移除
     * @param ElementPerson $elementPerson
     */
    public function remove(ElementPerson $elementPerson)
    {
        foreach ($this->elements as $key => $value) {
            if ($value == $elementPerson) {
                unset($this->elements[$key]);
            }
        }
    }

    /**
     * 查看显示
     * @param VisitorState $visitorState
     */
    public function display(VisitorState $visitorState)
    {
        foreach ($this->elements as $element) {
            $element->accept($visitorState);
        }
    }
}

// 客户端调用
$objectStructure = new ObjectStructure();
// 添加男人、女人
$objectStructure->add(new ConcreteElementMan());
$objectStructure->add(new ConcreteElementWoman());

// 成功时反应
$success = new ConcreteVisitorSuccess();
$objectStructure->display($success);

// 失败时反应
$failing = new ConcreteVisitorFailing();
$objectStructure->display($failing);

// 结果
男人:成功时,背后多半有一个伟大的女人。
女人:成功时,背后大多有一个不成功的男人。
男人:失败时,闷头喝酒,谁也不用劝。
女人:失败时,眼泪汪汪,谁也劝不了。

总结:

  • 访问者模式适用于数据结构相对稳定的系统,它把数据结构和作用于结构上的操作之间的耦合解脱开,使得操作集合可以相对自由地演化。
  • 访问者模式的目的是要把处理从数据结构分离出来。
  • 访问者模式的优点就是增加新的操作很容易,因为增加新的操作就意味着增加一个新的访问者,访问者模式将有关的行为集中到一个访问者对象中。缺点其实也就是使增加新的数据结构变得困难了。
(0)

相关推荐

  • C#设计模式学习笔记:(21)访问者模式

    本笔记摘抄自:https://www.cnblogs.com/PatrickLiu/p/8135083.html,记录一下学习过程以备后续查用. 一.引言 今天我们要讲行为型设计模式的第九个模式--访 ...

  • 诚之和:设计模式之什么是访问者模式

    本篇内容介绍了"设计模式之什么是访问者模式"的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学 ...

  • 设计模式(十六)——访问者模式

    设计模式(十六)——访问者模式

  • 访问者模式

    假设有男人和女人两种元素,要分别打印出他们在不同状态时的不同表现. 用OO的思想把表现(行为)提取出来作为一个抽象方法,代码如下: 用if-else对状态进行判断  Person接口 public i ...

  • 软件设计模式修炼 -- 访问者模式

    访问者模式是一种较为复杂的行为型设计模式,它包含访问者和被访问元素两个主要组成部分,这些被访问的元素具有不同的类型,且不同的访问者可以对其进行不同的访问操作 模式动机 对于系统中某些对象,它们存储在同 ...

  • 前置和后置操作

    前置和后置操作指的是在执行某个操作方法之前和之后会自动调用的方法,不过仅对访问控制器有效. 其他的分层控制器层和内部调用控制器的情况下前置和后置操作是无效的 系统会检测当前操作是否具有前置和后置操作, ...

  • 简说设计模式——访问者模式

    一.什么是访问者模式 访问者模式是一个相对比较简单,但结构又稍显复杂的模式,它讲的是表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作.例如,你在 ...

  • PHP设计模式之访问者模式

    PHP设计模式之访问者模式 访问者,就像我们去别人家访问,或者别人来我们家看望我们一样.我们每个人都像是一个实体,而来访的人都会一一的和我们打招呼.毕竟,我们中华民族是非常讲究礼数和好客的民族.访问者 ...

  • [PHP小课堂]PHP设计模式之访问者模式

    [PHP小课堂]PHP设计模式之访问者模式 关注公众号:[硬核项目经理]获取最新文章 添加微信/QQ好友:[DarkMatterZyCoder/149844827]免费得PHP.项目管理学习资料

  • 设计模式-行为型-访问者模式

    访问者模式(Vistor): 访问者模式的官方定义是这样的:表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变各元素类的前提下定义作用于这些元素的新操作.官方的东西总是晦涩难懂的,那么我们现 ...

  • 设计模式-门面模式

    门面模式 参考资料 图解设计模式 大话设计模式 设计模式之禅 github我见过最好的设计模式 http://c.biancheng.net/view/1326.html 定义 也称为外观模式 外观模 ...