C++ 设计模式:访问者模式(Visitor Pattern)

链接:C++ 设计模式
链接:C++ 设计模式 - 命令模式

访问者模式(Visitor Pattern)是一种行为型设计模式,它允许你在不改变元素类的前提下,定义作用于这些元素的新操作。访问者模式将操作的逻辑从元素类中分离出来,使得新的操作可以很容易地添加,而不需要修改元素类的代码。

1. 问题分析

在开发中,由于需求的改变,某些类层次结构经常需要增加新的行为(方法),如果直接在基类中做这样的更改,将会给子类带来很繁重的变更负担,甚至破坏原有设计,违反开闭原则(OCP)。

访问者模式通过将操作封装到访问者对象中,使得新的操作可以很容易地添加,而不需要修改元素类的代码。元素类只需要接受访问者,并将自己传递给访问者。

2.实现步骤

  1. 定义访问者接口(Visitor):声明对每个具体元素类的访问操作。
  2. 实现具体访问者类(ConcreteVisitor):实现访问者接口,定义具体的访问操作。
  3. 定义元素接口(Element):声明接受访问者的方法。
  4. 实现具体元素类(ConcreteElement):实现元素接口,定义接受访问者的方法,并将自己传递给访问者。
  5. 定义对象结构类(ObjectStructure):包含元素的集合,可以遍历这些元素并让访问者访问它们。

3.代码示例

3.1.定义访问者接口

cpp 复制代码
// 前向声明具体元素类
class ConcreteElementA;
class ConcreteElementB;

// 访问者接口
class Visitor {
 public:
  virtual ~Visitor() = default;
  virtual void visitConcreteElementA(ConcreteElementA* element) = 0;
  virtual void visitConcreteElementB(ConcreteElementB* element) = 0;
};

3.2.定义元素接口

cpp 复制代码
// 元素接口
class Element {
 public:
  virtual ~Element() = default;
  virtual void accept(Visitor* visitor) = 0;
};

3.3.实现具体元素类

cpp 复制代码
// 具体元素类A
class ConcreteElementA : public Element {
 public:
  void accept(Visitor* visitor) override { visitor->visitConcreteElementA(this); }
};
cpp 复制代码
// 具体元素类B
class ConcreteElementB : public Element {
 public:
  void accept(Visitor* visitor) override { visitor->visitConcreteElementB(this); }
};

3.4. 实现具体访问者类

cpp 复制代码
// 具体访问者类1:执行操作1
class ConcreteVisitor1 : public Visitor {
 public:
  void visitConcreteElementA(ConcreteElementA* element) override {
    std::cout << "ConcreteVisitor1: Processing ConcreteElementA" << std::endl;
  }

  void visitConcreteElementB(ConcreteElementB* element) override {
    std::cout << "ConcreteVisitor1: Processing ConcreteElementB" << std::endl;
  }
};
cpp 复制代码
// 具体访问者类2:执行操作2
class ConcreteVisitor2 : public Visitor {
 public:
  void visitConcreteElementA(ConcreteElementA* element) override {
    std::cout << "ConcreteVisitor2: Processing ConcreteElementA" << std::endl;
  }

  void visitConcreteElementB(ConcreteElementB* element) override {
    std::cout << "ConcreteVisitor2: Processing ConcreteElementB" << std::endl;
  }
};

3.5. 定义对象结构类

cpp 复制代码
// 对象结构类
class ObjectStructure {
 public:
  void addElement(Element* element) { elements_.push_back(element); }

  void accept(Visitor* visitor) {
    for (Element* element : elements_) {
      element->accept(visitor);
    }
  }

 private:
  std::vector<Element*> elements_;
};

3.6. 客户端代码

cpp 复制代码
int main() {
  // 创建对象结构
  ObjectStructure objectStructure;

  // 创建具体元素
  ConcreteElementA elementA;
  ConcreteElementB elementB;

  // 添加元素到对象结构
  objectStructure.addElement(&elementA);
  objectStructure.addElement(&elementB);

  // 创建具体访问者
  ConcreteVisitor1 visitor1;
  ConcreteVisitor2 visitor2;

  // 访问者访问对象结构中的元素
  std::cout << "Visitor1 visiting elements:" << std::endl;
  objectStructure.accept(&visitor1);

  std::cout << "Visitor2 visiting elements:" << std::endl;
  objectStructure.accept(&visitor2);

  return 0;
}

4.总结

  1. 优点
    • 扩展性:可以很容易地增加新的访问者,而无需修改元素类的代码。
    • 单一职责原则:将不同的操作分离到不同的访问者中,使得每个访问者只负责一种操作。
    • 开闭原则:元素类对扩展开放,对修改关闭。
  2. 缺点
    • 元素类的修改:每次增加新的元素时,都需要修改所有的访问者类,所以该模式的前提是元素类是稳定的。
    • 复杂性:如果元素类和访问者类较多,代码会变得复杂。
相关推荐
汉汉汉汉汉28 分钟前
C++11新特性详解:从列表初始化到线程库
c++
楼田莉子2 小时前
C++算法题目分享:二叉搜索树相关的习题
数据结构·c++·学习·算法·leetcode·面试
大锦终2 小时前
【算法】模拟专题
c++·算法
whitepure3 小时前
万字详解Java中的面向对象(二)——设计模式
java·设计模式
方传旺3 小时前
C++17 std::optional 深拷贝 vs 引用:unordered_map 查询大对象性能对比
c++
Dontla3 小时前
Makefile介绍(Makefile教程)(C/C++编译构建、自动化构建工具)
c语言·c++·自动化
何妨重温wdys3 小时前
矩阵链相乘的最少乘法次数(动态规划解法)
c++·算法·矩阵·动态规划
重启的码农3 小时前
ggml 介绍 (6) 后端 (ggml_backend)
c++·人工智能·神经网络
重启的码农3 小时前
ggml介绍 (7)后端缓冲区 (ggml_backend_buffer)
c++·人工智能·神经网络
雨落倾城夏未凉4 小时前
5.通过拷贝构造函数复制一个对象,假如对象的成员中有个指针类型的变量,如何避免拷贝出来的副本中的该成员之下行同一块内存(等价于默认拷贝构造函数有没有缺点)
c++·后端