【设计模式-访问者模式】

server/2024/10/4 20:25:27/

定义

访问者模式(Visitor Pattern)是一种行为型设计模式,允许你在不修改已有类的情况下向这些类添加新的功能或行为。它通过将操作的执行逻辑从对象的类中分离出来,使得你可以在保持类的封闭性(符合开闭原则)的前提下为不同对象定义新的操作。

UML图

在这里插入图片描述

  • Visitor(访问者):定义了针对元素结构中每种类型元素的访问操作,通常通过重载 visit 方法实现。
  • Element(元素接口或抽象类):定义了一个 accept(Visitor visitor) 方法,允许访问者访问自己。
  • ConcreteElement(具体元素):实现了 Element 接口,并在 accept 方法中调用访问者的对应方法,如 visitor.visit(this)。
  • ConcreteVisitor(对象结构):可以是一个包含不同类型元素的集合,它负责遍历元素并对每个元素调用 accept 方法。

优点

  • 开闭原则:可以在不修改现有类的情况下添加新的操作。
  • 单一职责原则:通过将元素的操作行为封装到访问者中,元素类的职责得以简化。
  • 可扩展性强:可以为对象结构中的类定义新的操作,而不改变类的定义。

缺点

  • 破坏封装性:访问者需要了解元素的内部细节,这可能破坏类的封装性。
  • 难以维护:如果元素类频繁变更,则需要更新所有访问者,访问者模式的维护成本可能较高。
  • 双重分派问题访问者模式的实现涉及双重分派,即通过 accept 方法调用访问者的 visit 方法,这使得结构变得复杂。

代码

// 定义访问者接口
interface Visitor {void visit(Book book);void visit(Fruit fruit);
}// 定义元素接口
interface ItemElement {void accept(Visitor visitor);
}// 具体元素类 Book
class Book implements ItemElement {private String title;private double price;public Book(String title, double price) {this.title = title;this.price = price;}public String getTitle() {return title;}public double getPrice() {return price;}@Overridepublic void accept(Visitor visitor) {visitor.visit(this);}
}// 具体元素类 Fruit
class Fruit implements ItemElement {private String name;private double weight;private double pricePerKg;public Fruit(String name, double weight, double pricePerKg) {this.name = name;this.weight = weight;this.pricePerKg = pricePerKg;}public String getName() {return name;}public double getWeight() {return weight;}public double getPricePerKg() {return pricePerKg;}@Overridepublic void accept(Visitor visitor) {visitor.visit(this);}
}// 实现具体的访问者
class ShoppingCartVisitor implements Visitor {@Overridepublic void visit(Book book) {System.out.println("Book: " + book.getTitle() + ", Price: " + book.getPrice());}@Overridepublic void visit(Fruit fruit) {double cost = fruit.getWeight() * fruit.getPricePerKg();System.out.println("Fruit: " + fruit.getName() + ", Cost: " + cost);}
}// 测试访问者模式
public class VisitorPatternDemo {public static void main(String[] args) {ItemElement[] items = new ItemElement[] {new Book("Design Patterns", 50),new Fruit("Apple", 2, 3)};Visitor visitor = new ShoppingCartVisitor();for (ItemElement item : items) {item.accept(visitor); // 访问每个元素}}
}

场景

  • 对象结构稳定:当你的对象结构(类)相对固定,但需要在这个结构上添加新的操作时,访问者模式非常合适。这样可以避免修改已有类的代码。
  • 需要对多个类执行相似的操作:当你需要对多个不同类型的元素执行类似的操作时,使用访问者模式可以集中管理这些操作。
  • 复杂的对象结构:在处理复杂的对象结构时(例如树形结构),访问者模式可以提供清晰的访问逻辑,避免在每个元素中实现相同的逻辑。
  • 数据结构和操作分离:当你希望将数据结构与操作分离,以便于未来扩展时,访问者模式是一个良好的选择。这样可以保持类的单一职责原则。
  • 频繁变化的操作:如果你的操作逻辑频繁变化,但元素结构相对稳定,使用访问者模式可以方便地添加新的操作而不影响已有的元素类。
  • 需要对元素进行类型判断访问者模式允许在访问者中根据元素的具体类型执行不同的操作,便于类型判断和特定处理。

具体示例

  • 编译器:在编译器中,抽象语法树(AST)需要执行不同的操作(如语义分析、优化等),访问者模式可以有效地管理这些操作。
  • 图形绘制:在图形编辑软件中,图形元素(如线条、圆形、矩形等)可以通过访问者模式实现不同的绘制和变换操作。
  • 文件系统:在文件系统中,可以定义访问者来实现文件的不同处理(如压缩、加密等),而不需要修改文件类的实现。
  • 账单处理:在购物车或账单系统中,可以使用访问者模式来处理不同类型的商品(如书籍、食品等),并计算总价或应用折扣。

总结

访问者模式适合于需要在多个元素上执行不同操作,且希望保持代码清晰和可扩展的情况。


http://www.ppmy.cn/server/122846.html

相关文章

《动手学深度学习》笔记2.1——神经网络从基础→进阶 (层和块 - 自定义块)

目录 0. 前言 原书正文(第五章) 第五章 - 第一节 - 层和块 - 自定义块 1. Sequential() PyTorch高级API 2. MLP() 无传入参数 3. MySequential() 传入任意层(块) 4. FixedHiddenMLP() 无传入参数-固定隐藏层 5. NestMLP() 传入嵌套块-多次嵌套 …

第七章 输入和输出处理 (IO流)

字符流:char Reader->InputStreamReader(字节流InputStream,编码)->FileReader(路径 | | FIle对象) 乱码问题:reader new InputStreamReader(new FileInputStream("路径"),"GBK"); import java.io.*;public class InputRe…

JAVA零基础入门——高级教程之集合框架

目录 1. 关于集合框架 1.1 集合接口 1.2 集合实现类 1.3 集合算法及迭代器和比较器的使用 2. 数据结构 2.1 ArrayList 2.2 LinkedList 2.3 HashMap 2.4 HashSet 3. 迭代器 1. 关于集合框架 集合框架是使用数据结构(参见本文2. 数据结构)来满…

【C++】红黑树的封装——同时实现map和set

目录 红黑树的完善默认成员函数迭代器的增加 红黑树的封装红黑树模板参数的控制仿函数解决取K问题对Key的非法操作 insert的调整map的[]运算符重载 在list模拟实现一文中,介绍了如何使用同一份代码封装出list的普通迭代器和const迭代器。今天学习STL中两个关联式容器…

[Linux]:信号(上)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 信号的引入 1.1 信号的概念 在Linux系统中,信号(…

【xhs截流软件】爬取小红书关键词笔记下的筛选评论

用python开发的xhs采集工具【爬小红书搜索评论软件】,可用于引流截流等。 支持2种模式的评论采集: 根据关键词采集评论,爬取思路:笔记关键词->笔记链接->评论根据笔记链接采集评论,爬取思路:笔记链接…

Could not find com.mapbox.mapboxsdk:mapbox-android-accounts:0.7.0.解决

AndroidStudio编译APK出现如下错误: Could not find com.mapbox.mapboxsdk:mapbox-android-accounts:0.7.0. 出现上面错误原因是因为没有打开对应的仓库导致的, 手动添加如下创建地址可解决: maven { url https://maven.aliyun.com/repos…

【python】while 循环的基础案例

目录: while 循环的基础语法while 循环的基础案例while 循环的嵌套应用while 循环的嵌套案例for 循环的基础语法for 循环的嵌套应用循环中断:break 和 continue综合案例 学习目标: 能够使用 while 循环,完成猜数字案例 猜数字…