灯火互联
管理员
管理员
  • 注册日期2011-07-27
  • 发帖数41778
  • QQ
  • 火币41290枚
  • 粉丝1086
  • 关注100
  • 终身成就奖
  • 最爱沙发
  • 忠实会员
  • 灌水天才奖
  • 贴图大师奖
  • 原创先锋奖
  • 特殊贡献奖
  • 宣传大使奖
  • 优秀斑竹奖
  • 社区明星
阅读:3062回复:0

[C++技术]设计模式C++描述----22.访问者(Visitor)模式

楼主#
更多 发布于:2012-09-06 11:34

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



二. 举例
假设有一项科学实验,是用来对比两种种子在不同环境下的生长情况。
两种种子,一种是普通的种子(Seed_A),一种是太空运回的种子(Seed_B)。
生长环境,分别是在多雨环境下(Rain_Status),阳光环境下(Sun_Status)等等。
结构如下:



代码如下:

[cpp] //状态  
class Status
{
public:
    virtual ~Status() {}
    
    virtual void VisitSeed_A(Seed* elm) {}

    virtual void VisitSeed_B(Seed* elm) {}

protected:
    Status() {}
};  

//下雨状态  
class Rain_Status:public Status
{
public:
    Rain_Status() {}
    
    virtual ~Rain_Status() {}
    
    //下雨状态下A种子生长的情况  
    virtual void VisitSeed_A(Seed* elm)
    {
        cout<<"Rain will visit Seed A..."<<endl;
    }
    
    //下雨状态下B种子生长的情况  
    virtual void VisitSeed_B(Seed* elm)
    {
        cout<<"Rain will visit Seed B..."<<endl;
    }
};  

//阳光状态  
class Sun_Status:public Status
{
public:
    Sun_Status() {}
    
    virtual ~Sun_Status() {}
    
    //阳光状态下A种子生长的情况  
    virtual void VisitSeed_A(Seed* elm)
    {
        cout<<"Sun will visit Seed A..."<<endl;
    }
    
    //阳光状态下B种子生长的情况  
    virtual void VisitSeed_B(Seed* elm)
    {
        cout<<"Sun will visit Seed B..."<<endl;
    }
};



//种子  
class Seed
{
public:
    virtual ~Seed() {}
    virtual void Accept(Status* vis) = 0;

protected:
    Seed() {}
};

//种子A,假设为普通种子  
class Seed_A:public Seed
{
public:
    Seed_A() {}
    
    ~Seed_A() {}
    
    void Accept(Status* vis)
    {
        vis->VisitSeed_A(this);
    }
};  

//种子B,假设为从太空带回来的种子  
class Seed_B:public Seed
{
public:
    Seed_B() {}
    ~Seed_B() {}
    
    void Accept(Status* vis)
    {
        vis->VisitSeed_B(this);
    }
};


//对象结构类,为了对比不同种子  
class ObjectStructure
{
private:
    list<Seed*> lseed;

public:
    //Add  
    void Attach(Seed* seed)
    {
        lseed.push_back(seed);
    }

    //Delete  
    void Detach(Seed* seed)
    {
        lseed.remove(seed);
    }

    //Show  
    void Display(Status* status)
    {
        list<Seed*>::iterator it = lseed.begin();
        
        for (it; it != lseed.end(); ++it)
        {
            (*it)->Accept(status);
        }
    }
};


//测试代码  
int main(int argc,char* argv[])
{
    ObjectStructure obj;
    
    //加入要对比的两个种子  
    obj.Attach(new Seed_A());
    obj.Attach(new Seed_B());

    //查看各种状态下两个种子的情况  
    obj.Display(new Rain_Status());
    
    //Sun Satte  
    obj.Display(new Sun_Status());

    return 0;
}
//状态
class Status
{
public:
    virtual ~Status() {}
  
    virtual void VisitSeed_A(Seed* elm) {}
    virtual void VisitSeed_B(Seed* elm) {}
protected:
    Status() {}
};
//下雨状态
class Rain_Status:public Status
{
public:
    Rain_Status() {}
  
    virtual ~Rain_Status() {}
  
    //下雨状态下A种子生长的情况
    virtual void VisitSeed_A(Seed* elm)
    {
        cout<<"Rain will visit Seed A..."<<endl;
    }
  
    //下雨状态下B种子生长的情况
    virtual void VisitSeed_B(Seed* elm)
    {
        cout<<"Rain will visit Seed B..."<<endl;
    }
};
//阳光状态
class Sun_Status:public Status
{
public:
    Sun_Status() {}
  
    virtual ~Sun_Status() {}
  
    //阳光状态下A种子生长的情况
    virtual void VisitSeed_A(Seed* elm)
    {
        cout<<"Sun will visit Seed A..."<<endl;
    }
  
    //阳光状态下B种子生长的情况
    virtual void VisitSeed_B(Seed* elm)
    {
        cout<<"Sun will visit Seed B..."<<endl;
    }
};

//种子
class Seed
{
public:
    virtual ~Seed() {}
    virtual void Accept(Status* vis) = 0;
protected:
    Seed() {}
};
//种子A,假设为普通种子
class Seed_A:public Seed
{
public:
    Seed_A() {}
  
    ~Seed_A() {}
  
    void Accept(Status* vis)
    {
        vis->VisitSeed_A(this);
    }
};
//种子B,假设为从太空带回来的种子
class Seed_B:public Seed
{
public:
    Seed_B() {}
    ~Seed_B() {}
  
    void Accept(Status* vis)
    {
        vis->VisitSeed_B(this);
    }
};

//对象结构类,为了对比不同种子
class ObjectStructure
{
private:
    list<Seed*> lseed;
public:
    //Add
    void Attach(Seed* seed)
    {
        lseed.push_back(seed);
    }
    //Delete
    void Detach(Seed* seed)
    {
        lseed.remove(seed);
    }
    //Show
    void Display(Status* status)
    {
        list<Seed*>::iterator it = lseed.begin();
      
        for (it; it != lseed.end(); ++it)
        {
            (*it)->Accept(status);
        }
    }
};

//测试代码
int main(int argc,char* argv[])
{
    ObjectStructure obj;
  
    //加入要对比的两个种子
    obj.Attach(new Seed_A());
    obj.Attach(new Seed_B());
    //查看各种状态下两个种子的情况
    obj.Display(new Rain_Status());
  
    //Sun Satte
    obj.Display(new Sun_Status());
    return 0;
}

三. 说明
1. 首先有一点要明确,就是两种种子不会轻易改变,也就是只有普通和太空种子两种。换句话说就是,数据结构比较稳定。
2. 可以变的是新增的状态,比如增加一个X光下的生成情况,等等。说白了就是,操作集合可以相对自由的演化。
3. 这种结构的优点是,增加新的操作很容易;缺点是,增加新的数据结构有点困难,因为你要在每一个访问者里都添加相应的操作。
4. 种子生长图相对于访问者模式的结构图有如下关系:
seed(种子)相当于 element(元素),这个是不怎么变的。

status(状态) 相当于 visitor(访问者),这个是可变且易变的。要注意的是,每个访问者都要对所有的元素(element)进行操作。
5. 事实上我们很少用这种模式,因为数据结构(element)不变的情况很少。


作者 lwbeyond


喜欢0 评分0
游客

返回顶部