《设计模式的艺术》笔记 - 访问者模式

介绍

        访问者模式提供一个作用于某对象结构中的各元素的操作表示,它使得可以在不改变各元素的类的前提下定义作用于这些元素的新操作。访问者模式是一种对象行为型模式。

实现

myclass.h

//
// Created by yuwp on 2024/1/12.
//

#ifndef DESIGNPATTERNS_MYCLASS_H
#define DESIGNPATTERNS_MYCLASS_H

#include <iostream>
#include <unordered_map>
#include <atomic>
#include <vector>
#include <memory>

class Visitor;

class Element { // 抽象元素类
public:
    virtual void accept(Visitor *visitor) = 0;
};

class ConcreteElementA : public Element {
public:
    void accept(Visitor *visitor) override;
};

class ConcreteElementB : public Element {
public:
    void accept(Visitor *visitor) override;
};

class Visitor {    // 抽象访问类
public:
    virtual void visit(ConcreteElementA *elementA) = 0;
    virtual void visit(ConcreteElementB *elementB) = 0;
};

class ConcreteVisitor : public Visitor {
public:
    void visit(ConcreteElementA *elementA) override;

    void visit(ConcreteElementB *elementB) override;
};

class ObjectStructure {
public:
    ~ObjectStructure();
    void addElement(Element *ele);
    void removeElement(Element *ele);
    void accept(Visitor *visitor);
private:
    std::vector<Element *> m_eles;
};

#endif //DESIGNPATTERNS_MYCLASS_H

myclass.cpp

//
// Created by yuwp on 2024/1/12.
//

#include "myclass.h"
#include <thread>
#include <unistd.h>
#include <sstream>

void ConcreteElementA::accept(Visitor *visitor) {
    std::cout << "A对象接受访问" << std::endl;
    visitor->visit(this);
}

void ConcreteElementB::accept(Visitor *visitor) {
    std::cout << "B对象接受访问" << std::endl;
    visitor->visit(this);
}

void ConcreteVisitor::visit(ConcreteElementA *elementA) {
    std::cout << "ConcreteVisitor访问了A对象" << std::endl;
}

void ConcreteVisitor::visit(ConcreteElementB *elementB) {
    std::cout << "ConcreteVisitor访问了B对象" << std::endl;
}

ObjectStructure::~ObjectStructure() {
    for (auto &e : m_eles) {
        if (e) {
            delete e;
        }
    }
}

void ObjectStructure::addElement(Element *ele) {
    m_eles.push_back(ele);
}

void ObjectStructure::removeElement(Element *ele) {
    for (auto it = m_eles.begin(); it != m_eles.end(); ++it) {
        if (*it == ele) {
            m_eles.erase(it);
            break;
        }
    }
}

void ObjectStructure::accept(Visitor *visitor) {
    for (auto &e : m_eles) {
        e->accept(visitor);
    }
}

main.cpp

#include <iostream>
#include <mutex>
#include "myclass.h"

int main() {
    ObjectStructure *objs = new ObjectStructure;
    objs->addElement(new ConcreteElementA);
    objs->addElement(new ConcreteElementB);
    Visitor *visitor = new ConcreteVisitor;
    objs->accept(visitor);

    delete objs;
    delete visitor;

    return 0;
}

总结

优点

        1. 增加新的访问操作很方便。使用访问者模式,增加新的访问操作就意味着增加一个新的具体访问者类,实现简单,无须修改源代码,符合开闭原则。

        2. 将有关元素对象的访问行为集中到一个访问者对象中,而不是分散在一个个的元素类中。类的职责更加清晰,有利于对象结构中元素对象的复用,相同的对象结构可以供多个不同的访问者访问。

        3. 让用户能够在不修改现有元素类层次结构的情况下,定义作用于该层次结构的操作。

缺点

        1. 增加新的元素类很困难。在访问者模式中,每增加一个新的元素类都意味着要在抽象访问者角色中增加一个新的抽象操作,并在每一个具体访问者类中增加相应的具体操作,这违背了开闭原则的要求。

        2. 破坏封装。访问者模式要求访问者对象访问并调用每一个元素对象的操作,这意味着元素对象有时候必须暴露一些自己的内部操作和内部状态,否则无法供访问者访问。

适用场景

        1. 一个对象结构包含多种类型的对象,希望对这些对象实施一些依赖其具体类型的操作。在访问者中针对每一种具体的类型都提供了一个访问操作,不同类型的对象可以有不同的访问操作。

        2. 需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而且需要避免让这些操作“污染”这些对象的类,也不希望在增加新操作时修改这些类。访问者模式将相关的访问操作集中起来定义在访问者类中,对象结构可以被多个不同的访问者类所使用,将对象本身与对象的访问操作分离。

        3. 对象结构中元素对象对应的类很少改变,但经常需要在此对象结构上定义新的操作。

练习

myclass.h

//
// Created by yuwp on 2024/1/12.
//

#ifndef DESIGNPATTERNS_MYCLASS_H
#define DESIGNPATTERNS_MYCLASS_H

#include <iostream>
#include <unordered_map>
#include <atomic>
#include <vector>
#include <memory>

class Visitor;

class Element { // 抽象元素类
public:
    Element(const std::string &name);
    virtual void accept(Visitor *visitor) = 0;

    uint32_t getMThesisNum() const;

    uint32_t getMScore() const;

    void setMThesisNum(uint32_t mThesisNum);

    void setMScore(uint32_t mScore);

    const std::string &getMName() const;

private:
    uint32_t m_thesisNum;
    uint32_t m_score;
    std::string m_name;
};

class Student : public Element {
public:
    Student(const std::string &name, uint32_t thesisNum, uint32_t score);
    void accept(Visitor *visitor) override;
};

class Teacher : public Element {
public:
    Teacher(const std::string &name, uint32_t thesisNum, uint32_t score);
    void accept(Visitor *visitor) override;
};

class Visitor {    // 抽象访问类
public:
    virtual void selection(Student *stu) = 0;
    virtual void selection(Teacher *tea) = 0;
};

class ScientificVisitor : public Visitor {
public:
    void selection(Student *stu) override;

    void selection(Teacher *tea) override;
};

class ExcellentVisitor : public Visitor {
public:
    void selection(Student *stu) override;

    void selection(Teacher *tea) override;
};

class ObjectStructure {
public:
    ~ObjectStructure();
    void addElement(Element *ele);
    void removeElement(Element *ele);
    void accept(Visitor *visitor);
private:
    std::vector<Element *> m_eles;
};

#endif //DESIGNPATTERNS_MYCLASS_H

myclass.cpp

//
// Created by yuwp on 2024/1/12.
//

#include "myclass.h"
#include <thread>
#include <unistd.h>
#include <sstream>

Element::Element(const std::string &name) {
    m_name = name;
}

uint32_t Element::getMThesisNum() const {
    return m_thesisNum;
}

uint32_t Element::getMScore() const {
    return m_score;
}

void Element::setMThesisNum(uint32_t mThesisNum) {
    m_thesisNum = mThesisNum;
}

void Element::setMScore(uint32_t mScore) {
    m_score = mScore;
}

const std::string &Element::getMName() const {
    return m_name;
}

Student::Student(const std::string &name, uint32_t thesisNum, uint32_t score) : Element(name) {
    setMThesisNum(thesisNum);
    setMScore(score);
}

void Student::accept(Visitor *visitor) {
    visitor->selection(this);
}

Teacher::Teacher(const std::string &name, uint32_t thesisNum, uint32_t score) : Element(name) {
    setMThesisNum(thesisNum);
    setMScore(score);
}

void Teacher::accept(Visitor *visitor) {
    visitor->selection(this);
}

ObjectStructure::~ObjectStructure() {
    for (auto &e : m_eles) {
        if (e) {
            delete e;
        }
    }
}

void ObjectStructure::addElement(Element *ele) {
    m_eles.push_back(ele);
}

void ObjectStructure::removeElement(Element *ele) {
    for (auto it = m_eles.begin(); it != m_eles.end(); ++it) {
        if (*it == ele) {
            m_eles.erase(it);
            break;
        }
    }
}

void ObjectStructure::accept(Visitor *visitor) {
    for (auto &e : m_eles) {
        e->accept(visitor);
    }
}


void ScientificVisitor::selection(Student *stu) {
    if (stu->getMThesisNum() >= 2) {
        std::cout << stu->getMName() << "同学可以被评为科研奖" << std::endl;
    } else {
        std::cout << stu->getMName() << "同学不能被评为科研奖" << std::endl;
    }
}

void ScientificVisitor::selection(Teacher *tea) {
    if (tea->getMThesisNum() >= 10) {
        std::cout << tea->getMName() << "老师可以被评为科研奖" << std::endl;
    } else {
        std::cout << tea->getMName() << "老师不能被评为科研奖" << std::endl;
    }
}

void ExcellentVisitor::selection(Student *stu) {
    if (stu->getMScore() >= 90) {
        std::cout << stu->getMName() << "同学可以被评为优秀奖" << std::endl;
    } else {
        std::cout << stu->getMName() << "同学不能被评为优秀奖" << std::endl;
    }
}

void ExcellentVisitor::selection(Teacher *tea) {
    if (tea->getMScore() >= 90) {
        std::cout << tea->getMName() << "老师可以被评为优秀奖" << std::endl;
    } else {
        std::cout << tea->getMName() << "老师不能被评为优秀奖" << std::endl;
    }
}

main.cpp

#include <iostream>
#include <mutex>
#include "myclass.h"

int main() {
    ObjectStructure *objs = new ObjectStructure;
    objs->addElement(new Student("小明", 2, 80));
    objs->addElement(new Teacher("老王", 8, 95));
    Visitor *visitor = new ScientificVisitor;
    objs->accept(visitor);
    delete visitor;
    visitor = new ExcellentVisitor;
    objs->accept(visitor);
    delete objs;
    delete visitor;

    return 0;
}

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

vfdn

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值