Visitor模式与Acyclic Visitor

原创 2007年09月19日 15:42:00

visitor模式最基本的是访问者和被访问者。它的优势是不用改变被访问类的任何结构就可以对被访问者施加新的操作,前提是增加一个访问者的子类。其实就是用增加访问者子类的方法替代被访问者方法的增加。

用你不需要的便利换取你需要的功能。

访问者执行一些对被访问者的一些操作,实现方式是在被访问者中调用访问者的方法,把自己(this指针)传进去。这样对被访问者的一次调用被隐式转移到访问者对被访问者的一些处理。但这样有问题,被访问者要知道访问者基类,访问者要知道所有的被访问者子类。 

Acyclic Visitor做的事情是:

Visitor基类只是一个基类(稻草人),它对一切都无所知。但是被访问者要知道Visitor的继承者,也就是真是的访问者,被访问者依据dynamic_cast得到具体的visitor,然后调用visitor的定制方法。这个变种模式使得visitor不需要知道任何visited,只有visited单向的需要知道visitor。

但是,特定的visited的访问者需要知道这个特定的visited。所以这个问题的解决方案可以用代码简单描述为:

//f1    visitor base file
class Visitor
...{
    
virtual    void    ~Visitor();
}
;

//f2    visitor-A file
class A;
class VisitorA    :public Visitor
...{
public:
    
void    DoVisitA(A& a)...{}
}
;

//f3    visitor-B file
class B;
class VisitorB    :public Visitor
...{
public:
    
void    DoVisitB(B& b)...{}
}
;

//f4    class A file
class VisitorA;
class A    :public Base
...{
public:
    
void    Accept(Visitor& v)
    
...{
        
if (VisitorA& va = dynamic_cast<VisitorA>(v))
        
...{
            va.DoVisitA(
*this);
        }

        
else
        
...{
            
//default visit operator
        }

    }

}
;

//f5    class B file
class VisitorB;
class B :public Base
...{
public:
    
void    Accept(Visitor& v)
    
...{
        
if (VisitorB& vb = dynamic_cast<VisitorB>(v))
        
...{
            vb.DoVisitB(
*this);
        }

        
else
        
...{
            
//default visit operator
        }

    }

}
;

//f6    concret visitor
//#include class A,class B
class ConcretVisitor
    :
public    VisitorA,public VisitorB
...{
public:
    
void    VisitorA();    //do process self data
    void    VisitorB();    //do process self data
}
;

//while use
ConcretVisitor    mv;
A                ma;
B                mb;
ma.Accept(mv);
mb.Accept(mv);
mv.
out-put-self-data;

fn代表第n个文件,可以看到,特定的visitor只需知道与自己对于的visited就可以了。而visited某种程度上也是这样。只不过,ConcretVisitor需要知道一切,这是肯定的,因为它是终端用户。这个方案很好解决了循环引用的问题,所以这叫Acyclic(非循环的)。

请注意一个名词:名称上的依存。这意味着,虽然是依存,但这仅仅需要前置声明就可以,不必须包含文件。

Loki库里的Acyclic Visitor

已经知道,使用 Visitor有这样那样的不便。Loki为了避免这些提供了两个方案,之一就是Acyclic的类库化。class BaseVisitor...{public:    virtual   ...
  • win2ks
  • win2ks
  • 2007年09月19日 16:16
  • 667

再议访问者模式 - Visitor vs Acyclic Visitor

传统访问者模式 VS 无环访问者模式 (ACyclic Visitor)
  • zj510
  • zj510
  • 2014年09月16日 15:23
  • 1761

设计模式(行为型)之访问者模式(Visitor Pattern)

访问者模式是一种较为复杂的行为型设计模式,它包含访问者和被访问元素两个主要组成部分,这些被访问的元素通常具有不同的类型,且不同的访问者可以对它们进行不同的访问操作。访问者模式使得用户可以在不修改现有系...
  • yanbober
  • yanbober
  • 2015年05月06日 17:11
  • 3226

JAVA设计模式之Visitor模式

一个集合(Collection)中,可以包含一个Car,也可以包含一个Cat,对于不同类型的元素,他们的行为也不尽相同,比如,Car可能有start()行为,而Cat可能有eat()的行为。可是对于C...
  • chenjie19891104
  • chenjie19891104
  • 2011年05月04日 13:46
  • 13496

Java设计模式(三) Visitor(访问者)模式及多分派场景应用

基本概念Visitor 封装一些作用于数据结构中的各元素的操作,不同的操作可以借助新的visitor实现,降低了操作间的耦合性 访问者可以将数据结构和对数据的操作解耦,使得增加对数据结构的操作不需要取...
  • qq_24451605
  • qq_24451605
  • 2016年04月14日 15:32
  • 5194

JAVA设计模式十四--Visitor(访问者模式)

访问者模式(Visitor Pattern)是GoF提出的23种设计模式中的一种,属于行为模式。 据《大话设计模式》中说算是最复杂也是最难以理解的一种模式了。    定义(源于GoF《Design...
  • hfmbook
  • hfmbook
  • 2012年06月22日 10:20
  • 8925

设计模式总结之Visitor Pattern(访问者模式)

表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。...
  • cooldragon
  • cooldragon
  • 2016年08月12日 12:10
  • 4103

设计模式:Visitor模式

Visitor模式是一个用起来很简单,理解起来可能稍微有一点困难的模式。不过明白了之后就清楚了,其实也是非常的简单。问题需要向对象结构中增加新的方法,但是增加起来会很费劲或者会破坏设计。 案例举一个例...
  • superbeck
  • superbeck
  • 2010年02月25日 14:02
  • 10969

设计模式:VISITOR模式

最近在项目中用到了VISITOR模式,总结一下,自己也再学习一遍,同时和大家分享。 我最初遇到设计模式的时候,有这些疑问:什么设计模式在什么情况下可以解决什么问题?设计模式的最大特点是抽象,并不难,...
  • u012142787
  • u012142787
  • 2014年09月28日 20:09
  • 967

利用dom4j的visitor模式对xml文件的“反射”

给定一个xml文件,由于dom4j支持visitor模式,可以得到生成该xml文件内容的java代码(dom4j的API),在某些时候,可能会有用处。比如给定一个行数超过50行的范例xml文件,其内容...
  • luocm
  • luocm
  • 2007年08月20日 14:54
  • 1941
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Visitor模式与Acyclic Visitor
举报原因:
原因补充:

(最多只允许输入30个字)