常见设计模式的解析和实现(C++)之二十-Visitor模式
表示一个作用于某对象结构中的各元素的操作.它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作.
UML结构图:
解析:
Visitor模式把对结点的访问封装成一个抽象基类,通过派生出不同的类生成新的访问方式.在实现的时候,在visitor抽象基类中声明了对所有不同结点进行访问的接口函数,如图中的VisitConcreateElementA函数等,这样也造成了Visitor模式的一个缺陷--新加入一个结点的时候都要添加Visitor中的对其进行访问接口函数,这样使得所有的Visitor及其派生类都要重新编译了,也就是说Visitor模式一个缺点就是添加新的结点十分困难.另外,还需要指出的是Visitor模式采用了所谓的"双重分派"的技术,拿上图来作为例子,要对某一个结点进行访问,首先需要产生一个Element的派生类对象,其次要传入一个Visitor类派生类对象来调用对应的Accept函数,也就是说,到底对哪种Element采用哪种Visitor访问,需要两次动态绑定才可以确定下来,具体的实现可以参考下面实现代码中的Main.cpp部分是如何调用这些类的.
实现:
1)Visitor.h
/********************************************************************
created: 2006/08/09
filename: Visitor.h
author: 李创
http://www.cppblog.com/converse/
purpose: Visitor模式的演示代码
*********************************************************************/

#ifndef VISITOR_H
#define
VISITOR_H

class
Visitor;
class
Element
{
public:
virtual ~Element(){}
virtual void Accept(Visitor &rVisitor) = 0;
protected:
Element(){}
}
;
class
ConcreateElementA
:
public
Element
{
public:
virtual ~ConcreateElementA() {}
virtual void Accept(Visitor &rVisitor);
}
;
class
ConcreateElementB
:
public
Element
{
public:
virtual ~ConcreateElementB() {}
virtual void Accept(Visitor &rVisitor);
}
;
class
Visitor
{
public:
virtual ~Visitor(){}
virtual void VisitConcreateElementA(ConcreateElementA *pConcreateElementA) = 0;
virtual void VisitConcreateElementB(ConcreateElementB *pConcreateElementB) = 0;
protected:
Visitor(){}
}
;
class
ConcreateVisitorA
:
public
Visitor
{
public:
virtual ~ConcreateVisitorA(){}
virtual void VisitConcreateElementA(ConcreateElementA *pConcreateElementA);
virtual void VisitConcreateElementB(ConcreateElementB *pConcreateElementB);
}
;
class
ConcreateVisitorB
:
public
Visitor
{
public:
virtual ~ConcreateVisitorB(){}
virtual void VisitConcreateElementA(ConcreateElementA *pConcreateElementA);
virtual void VisitConcreateElementB(ConcreateElementB *pConcreateElementB);
}
;
#endif
2)Visitor.cpp
/********************************************************************
created: 2006/08/09
filename: Visitor.cpp
author: 李创
http://www.cppblog.com/converse/
purpose: Visitor模式的演示代码
*********************************************************************/

#include
"
Visitor.h
"
#include
<
iostream
>

void
ConcreateElementA::Accept(Visitor
&
rVisitor)
{
rVisitor.VisitConcreateElementA(this);
}

void
ConcreateElementB::Accept(Visitor
&
rVisitor)
{
rVisitor.VisitConcreateElementB(this);
}

void
ConcreateVisitorA::VisitConcreateElementA(ConcreateElementA
*
pConcreateElementA)
{
std::cout << "VisitConcreateElementA By ConcreateVisitorA/n";
}

void
ConcreateVisitorA::VisitConcreateElementB(ConcreateElementB
*
pConcreateElementA)
{
std::cout << "VisitConcreateElementB By ConcreateVisitorA/n";
}

void
ConcreateVisitorB::VisitConcreateElementA(ConcreateElementA
*
pConcreateElementA)
{
std::cout << "VisitConcreateElementA By ConcreateVisitorB/n";
}

void
ConcreateVisitorB::VisitConcreateElementB(ConcreateElementB
*
pConcreateElementA)
{
std::cout << "VisitConcreateElementB By ConcreateVisitorB/n";
}
3)Main.cpp
/********************************************************************
created: 2006/08/09
filename: Main.cpp
author: 李创
http://www.cppblog.com/converse/
purpose: Visitor模式的测试代码
*********************************************************************/
#include "Visitor.h"
int main()
{
Visitor *pVisitorA = new ConcreateVisitorA();
Element *pElement = new ConcreateElementA();
pElement->Accept(*pVisitorA);
delete pElement;
delete pVisitorA;
return 0;
}
访问者模式,不改变被元素类情况下,通过改变访问者类接口来实现不同的访问方式。
本文介绍C++中Visitor模式的设计原理及其实现方法。通过该模式可在不修改元素类的前提下定义新操作,适用于需要对对象结构中的元素进行操作的场景。

690

被折叠的 条评论
为什么被折叠?



