参观者模式
声明提示:本模式是参考《设计模式》gof 中的模式。
1. 意图
表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。
2.适用性
在下列情况下使用Visitor模式:
• 一个对象结构包含很多类对象,它们有不同的接口,而你想对这些对象实施一些依赖于其具体类的操作。
• 需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而你想避免让这些操作“污染”这些对象的类。 Visitor使得你可以将相关的操作集中起来定义在一个类中。当该对象结构被很多应用共享时,用 Visitor模式让每个应用仅包含需要用到的操作。
• 定义对象结构的类很少改变,但经常需要在此结构上定义新的操作。改变对象结构类需要重定义对所有访问者的接口,这可能需要很大的代价。如果对象结构类经常改变,那
么可能还是在这些类中定义这些操作较好。
3.类图
4.参与者
• Visitor(访问者,如NodeVisitor)
— 为该对象结构中 ConcreteElement的每一个类声明一个 Visit操作。该操作的名字和特征标识了发送 Visit请求给该访问者的那个类。这使得访问者可以确定正被访问元素
的具体的类。这样访问者就可以通过该元素的特定接口直接访问它。
• ConcreteVisitor(具体访问者,如TypeCheckingVisitor)
— 实现每个由 Visitor声明的操作。每个操作实现本算法的一部分,而该算法片断乃是对应于结构中对象的类。 ConcreteVisitor为该算法提供了上下文并存储它的局部状态。这一状态常常在遍历该结构的过程中累积结果。
• Element(元素,如Node)
— 定义一个Accept操作,它以一个访问者为参数。
• ConcreteElement(具体元素,如AssignmentNode, VariableRefNode)
— 实现Accept操作,该操作以一个访问者为参数。
• ObjectStructure(对象结构,如Program)
— 能枚举它的元素。
— 可以提供一个高层的接口以允许该访问者访问它的元素。
— 可以是一个复合(参见 Composite)或是一个集合,如一个列表或一个无序集合。
5.代码:
#pragma once
#include "stdafx.h"
class Element;
class Visitor
{
public:
virtual ~Visitor() {}
virtual void visterConcreatElementA(Element * elementA) = 0;
virtual void visterConcreatElementB(Element * elementB) = 0;
protected:
Visitor() {};
};
class Element
{
public:
Element() {}
virtual ~Element() {}
virtual void Accpet(Visitor * vis) = 0;
virtual void show() = 0;
};
class ConcreateVisitor1 : public Visitor
{
public:
ConcreateVisitor1() {}
~ConcreateVisitor1() {}
virtual void visterConcreatElementA(Element * elementA)
{
cout << "visterConcreatElementA" << endl;
elementA->show();
}
virtual void visterConcreatElementB(Element * elementB)
{
cout << "visterConcreatElementB" << endl;
elementB->show();
}
};
class ConcreateVisitor2 : public Visitor
{
public:
ConcreateVisitor2() {}
~ConcreateVisitor2() {}
virtual void visterConcreatElementA(Element * elementA)
{
cout << "visterConcreatElementA" << endl;
elementA->show();
}
virtual void visterConcreatElementB(Element * elementB)
{
cout << "visterConcreatElementB" << endl;
elementB->show();
}
};
class ConcreatElementA : public Element
{
public:
ConcreatElementA(){}
~ConcreatElementA() {}
virtual void Accpet(Visitor * vis)
{
vis->visterConcreatElementA(this);
}
virtual void show()
{
cout << "ConcreatElementA show" << endl;
}
};
class ConcreatElementB : public Element
{
public:
ConcreatElementB() {}
~ConcreatElementB() {}
virtual void Accpet(Visitor * vis)
{
vis->visterConcreatElementB(this);
}
virtual void show()
{
cout << "ConcreatElementB show" << endl;
}
};
class ObjectStructure
{
public:
//add 数据
void Attach(Element * ele)
{
auto it = find_if(m_list.begin(), m_list.end(),
[ele](Element * ptr) {return ptr == ele; });
if (it == m_list.end())
{
m_list.push_back(ele);
}
}
void Detach(Element * ele)
{
auto it = find_if(m_list.begin(), m_list.end(),
[ele](Element * ptr) {return ptr == ele; });
if (it != m_list.end())
{
delete *it;
m_list.remove(ele);
}
}
void Display(Visitor * vis)
{
for (auto it : m_list)
{
it->Accpet(vis);
}
}
private:
list<Element *> m_list;
};
void process()
{
ObjectStructure *obj = new ObjectStructure;
//添加两个元素
obj->Attach(new ConcreatElementA());
obj->Attach(new ConcreatElementB());
//查看对应的参观
obj->Display(new ConcreateVisitor1());
obj->Display(new ConcreateVisitor2());
}
stdafx.h
#include <iostream>
#include <list>
#include <algorithm>
using namespace std;
输出结果:
visterConcreatElementA
ConcreatElementA show
visterConcreatElementB
ConcreatElementB show
visterConcreatElementA
ConcreatElementA show
visterConcreatElementB
ConcreatElementB show
喜欢我的博客,可以关注点赞,让咱们更快的学习,一起加油。