设计模式(23)-访问者模式


访问者模式

简介

表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变各元素的类的前提下
定义作用于这些元素的新操作

访问者模式适用于数据结构相对稳定的系统
它把数据结构和作用于结构上的操作之间的耦合脱开,使得操作合集可以相对自由地演化

访问者模式的目的是要把处理从数据结构中分离出来
如果系统有比较稳定的数据结构,又有易于变化的算法的话,使用访问者模式就是比较适合的,
因为访问者模式使得算法操作的增加变得容易

访问者模式的优点就是增加新的操作很容易,因为增加新的操作就意味着增加一个新的访问者,
访问者模式将有关的行为集中到一个访问者对象中
访问者模式的缺点其实也就是使增加新的数据结构变得困难了

代码示例

<?php
/**
 * 为该对象结构中的 ConcreteElement 的每一个类声明一个 Visit 操作
 * Class Visitor
 */
abstract class Visitor
{
    // 总的数据
    abstract public function visitConcreteElementA(ConcreteElementA $concreteElementA);

    abstract public function visitConcreteElementB(ConcreteElementB $concreteElementB);
}

/**
 * 具体访问者,实现每个由 Visitor 声明的操作。
 * 每个操作实现算法的一部分,而该算法片段乃是对应于结构中对象的类
 *
 * Class ConcreteVisitor1
 */
class ConcreteVisitor1 extends Visitor
{
    public function visitConcreteElementA(ConcreteElementA $concreteElementA)
    {
        echo get_class($concreteElementA) . '被' . get_class($this) . '访问' . PHP_EOL;
    }

    public function visitConcreteElementB(ConcreteElementB $concreteElementB)
    {
        echo get_class($concreteElementB) . '被' . get_class($this) . '访问' . PHP_EOL;
    }
}

class ConcreteVisitor2 extends Visitor
{
    public function visitConcreteElementA(ConcreteElementA $concreteElementA)
    {
        echo get_class($concreteElementA) . '被' . get_class($this) . '访问' . PHP_EOL;
    }

    public function visitConcreteElementB(ConcreteElementB $concreteElementB)
    {
        echo get_class($concreteElementB) . '被' . get_class($this) . '访问' . PHP_EOL;
    }
}

/**
 * 定义一个 accept 操作,它以一个访问者为参数
 *
 * Class Element
 */
abstract class Element
{
    abstract public function accept(Visitor $visitor);
}

class ConcreteElementA extends Element
{
    public function accept(Visitor $visitor)
    {
        $visitor->visitConcreteElementA($this);
    }

    public function operationA()
    {

    }
}

class ConcreteElementB extends Element
{
    public function accept(Visitor $visitor)
    {
        $visitor->visitConcreteElementB($this);
    }

    public function operationB()
    {

    }
}

/**
 * 能枚举它的元素,可以提供一个高层的接口以允许访问者访问它的元素
 *
 * Class ObjectStructure
 */
class ObjectStructure
{
    /**
     * @var Element[]
     */
    private $elements;

    public function attach(Element $element)
    {
        $key = get_class($element);
        $this->elements[$key] = $element;
    }

    public function detach(Element $element)
    {
        $key = get_class($element);
        if (isset($this->elements[$key])) {
            unset($this->elements[$key]);
        }
    }

    public function accept(Visitor $visitor)
    {
        foreach ($this->elements as $v) {
            $v->accept($visitor);
        }
    }
}

$o = new ObjectStructure();

// 添加要被访问的元素
$o->attach(new ConcreteElementA());
$o->attach(new ConcreteElementB());

// 每个访问者循环访问元素
$o->accept(new ConcreteVisitor1());
$o->accept(new ConcreteVisitor2());

文章作者: 江湖义气
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 江湖义气 !
  目录