【设计模式】【行为型模式】访问者模式(Visitor)

👋hi,我不是一名外包公司的员工,也不会偷吃茶水间的零食,我的梦想是能写高端CRUD
🔥 2025本人正在沉淀中… 博客更新速度++
👍 欢迎点赞、收藏、关注,跟上我的更新节奏
🎵 当你的天空突然下了大雨,那是我在为你炸乌云

文章目录

  • 一、入门
    • 什么是访问者模式?
    • 为什么需要访问者模式?
    • 怎么实现访问模式?
  • 二、访问者模式在源码中的运用
    • ASM 框架(Java 字节码操作)
  • 三、总结
    • 访问者模式的优点
    • 访问者模式的缺点
    • 访问者模式的适用场景

一、入门

什么是访问者模式?

访问者模式(Visitor Pattern)是一种行为设计模式,允许你将算法与对象结构分离。通过这种方式,可以在不改变对象结构的情况下,向对象结构中的元素添加新的操作。

为什么需要访问者模式?

假设有一个图形对象结构,包含CircleRectangle两种元素,需要实现两种操作:计算面积和导出为JSON。
传统实现(无访问者模式):

// 元素类
interface Shape {double calculateArea();  // 操作1:计算面积String toJson();        // 操作2:导出为JSON
}class Circle implements Shape {@Overridepublic double calculateArea() { /* 实现 */ }@Overridepublic String toJson() { /* 实现 */ }
}class Rectangle implements Shape {@Overridepublic double calculateArea() { /* 实现 */ }@Overridepublic String toJson() { /* 实现 */ }
}

问题

  1. 违反开闭原则
    • 当需要为对象结构添加新操作时(例如计算、导出、校验等),必须修改每个元素类的代码。
    • 示例:每新增一个操作(如导出为XML),所有Shape子类都需要修改。
  2. 代码冗余和分散
    • 相关操作分散在各个元素类中,难以集中管理。
    • 示例:如果“校验”逻辑分散在Circle.validate()Rectangle.validate()中,维护和扩展会变得困难。
  3. 难以扩展复杂操作
    • 某些操作需要跨多个元素协作(例如统计图形的面积总合),直接写在元素类中会导致职责混乱。

怎么实现访问模式?

访问者模式的构成如下

  1. Visitor(访问者):定义了对每个元素(Element)的访问操作,通常为每个具体元素类提供一个访问方法。
  2. ConcreteVisitor(具体访问者):实现Visitor接口,定义具体的操作。
  3. Element(元素):定义一个接受访问者的方法(accept),通常是一个接口或抽象类。
  4. ConcreteElement(具体元素):实现Element接口,提供具体的accept方法实现。
  5. ObjectStructure(对象结构):包含一组元素,通常提供一个方法让访问者访问其中的所有元素。

【案例】图形对象结构 - 改
在这里插入图片描述
Visitor(访问者接口):对应 ShapeVisitor 接口
作用:定义访问操作的接口,声明对不同具体元素(如CircleRectangle)的访问方法。

interface ShapeVisitor {void visit(Circle circle);    // 访问 Circle 元素void visit(Rectangle rectangle); // 访问 Rectangle 元素
}

ConcreteVisitor(具体访问者):对应AreaCalculatorJsonExporter
作用:实现 ShapeVisitor 接口,定义具体的操作逻辑(如计算面积、导出JSON)。

// 具体访问者1:计算面积
class AreaCalculator implements ShapeVisitor {@Overridepublic void visit(Circle circle) { /* 计算圆的面积 */ }@Overridepublic void visit(Rectangle rectangle) { /* 计算矩形的面积 */ }
}// 具体访问者2:导出为JSON
class JsonExporter implements ShapeVisitor {@Overridepublic void visit(Circle circle) { /* 导出圆的JSON */ }@Overridepublic void visit(Rectangle rectangle) { /* 导出矩形的JSON */ }
}

Element(元素接口):对应Shape接口
作用:定义元素的通用行为,即通过accept方法接受访问者。

interface Shape {void accept(ShapeVisitor visitor); // 接受访问者的入口
}

ConcreteElement(具体元素):对应CircleRectangle
作用:实现Shape接口,在accept方法中将自身传递给访问者的具体方法(如visit(Circle))。

class Circle implements Shape {@Overridepublic void accept(ShapeVisitor visitor) {visitor.visit(this); // 调用访问者的 visit(Circle) 方法}
}class Rectangle implements Shape {@Overridepublic void accept(ShapeVisitor visitor) {visitor.visit(this); // 调用访问者的 visit(Rectangle) 方法}
}

ObjectStructure(对象结构):通常是一个管理元素集合的类。
作用:负责维护一组元素(如Shape对象),并提供遍历方法让访问者访问所有元素。

class ShapeCollection {private List<Shape> shapes = new ArrayList<>();public void addShape(Shape shape) {shapes.add(shape);}// 让访问者遍历所有元素public void accept(ShapeVisitor visitor) {for (Shape shape : shapes) {shape.accept(visitor);}}
}

二、访问者模式在源码中的运用

ASM 框架(Java 字节码操作)

ASM 框架是一个用于操作 Java 字节码的库,广泛用于动态生成类、修改类文件(如 AOP、代码增强)等场景。它通过访问者模式(Visitor Pattern)将字节码的解析和生成与具体操作解耦。
ASM 通过以下步骤实现字节码操作:

  1. 解析字节码:ClassReader 读取 .class 文件。
  2. 触发访问者:ClassReader 将字节码中的每个元素(类、方法、字段等)传递给 ClassVisitr
  3. 处理元素:开发者通过自定义的 ClassVisitorMethodVisitor 修改或分析字节码。
  4. 生成新字节码:ClassWriter 将修改后的字节码写入新的 .class 文件。

访问者模式角色 ASM 中的实现:

  1. VisitorClassVisitorMethodVisitorFieldVisitor等接口
  2. ConcreteVisitor: 开发者自定义的ClassVisitorMethodVisitor实现
  3. Element: 字节码中的结构(类、方法、字段、指令等)
  4. ConcreteElement: 具体的类、方法、字段(如 visitMethod中的方法描述符)
  5. ObjectStructureClassReader(负责遍历类文件并触发访问者方法)

Visitor: 以ClassVisitor为例子,它的作用是,定义访问类结构的操作,例如访问类头、方法、字段等。

public abstract class ClassVisitor {protected final int api;protected ClassVisitor cv;...public void visit(final int version,final int access,final String name,final String signature,final String superName,final String[] interfaces) {if (api < Opcodes.ASM8 && (access & Opcodes.ACC_RECORD) != 0) {throw new UnsupportedOperationException("Records requires ASM8");}if (cv != null) {cv.visit(version, access, name, signature, superName, interfaces);}}...
ConcreteVisitor:我们用户自己实现的Visitor接口的类
public class TimerClassVisitor extends ClassVisitor {public TimerClassVisitor(ClassVisitor cv) {super(Opcodes.ASM9, cv);}@Overridepublic MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);// 为所有非构造方法添加计时逻辑if (!name.equals("<init>")) {mv = new MethodTimerVisitor(mv, name);}return mv;}
}

Element:在 ASM 框架中,Element的概念是隐式的,而不是显式地通过一个接口或类来表示。字节码中的各种结构(如类、方法、字段、指令等)可以被视为Element,但它们并没有一个统一的接口或基类。相反,ASM 通过访问者模式直接操作这些结构。

在 ASM 中,以下结构可以被视为Element

  1. :通过 ClassVisitor.visit 方法的参数(如类名、父类、接口等)表示。
  2. 方法:通过 ClassVisitor.visitMethod 方法的参数(如方法名、描述符等)表示。
  3. 字段:通过 ClassVisitor.visitField 方法的参数(如字段名、类型等)表示。
  4. 指令:通过 MethodVisitor.visitInsnvisitMethodInsn 等方法的参数表示。

ConcreteElement :在 ASM 框架中,ConcreteElement并不是通过一个显式的类或接口来表示的,而是通过访问者模式的方法参数隐式传递的。具体来说,字节码中的各种结构(如类、方法、字段、指令等)可以被视为ConcreteElement,但它们并没有一个统一的基类或接口。
ClassVisitor.visit方法中,类的信息通过参数传递:

void visit(int version,              // 类版本int access,               // 访问标志(如 public、final)String name,              // 类名String signature,         // 泛型签名String superName,         // 父类名String[] interfaces       // 实现的接口
);

ObjectStructure:在 ASM 框架中,ClassReader就是这个角色,它负责读取类文件并触发访问者模式的方法调用。
以下是accept方法的简化伪代码,展示了ClassReader如何触发访问者方法:

public void accept(ClassVisitor cv, int parsingOptions) {// 解析类头cv.visit(version, access, name, signature, superName, interfaces);// 解析字段for (FieldInfo field : fields) {cv.visitField(field.access, field.name, field.desc, field.signature, field.value);}// 解析方法for (MethodInfo method : methods) {MethodVisitor mv = cv.visitMethod(method.access, method.name, method.desc, method.signature, method.exceptions);if (mv != null) {// 解析方法体for (Instruction insn : method.instructions) {mv.visitInsn(insn.opcode);}mv.visitEnd();}}// 结束访问cv.visitEnd();
}

三、总结

访问者模式的优点

  1. 开闭原则
    • 优点:新增操作时只需添加新的访问者类,无需修改现有的对象结构。
    • 示例:如果需要为类结构添加新的操作(如导出为XML),只需实现一个新的访问者类。
  2. 单一职责原则
    • 优点:将相关操作集中在一个访问者类中,便于维护和扩展。
    • 示例:将“计算面积”和“导出为JSON”的逻辑分别放在不同的访问者类中。
  3. 灵活性
    • 优点:可以在不修改对象结构的情况下,动态地为对象结构添加新的操作。
    • 示例:在编译器中使用访问者模式实现语法树的遍历和优化。
  4. 解耦数据结构与操作
    • 优点:将数据结构与操作逻辑分离,使得代码更清晰、更易于维护。
    • 示例:ASM 框架通过访问者模式将字节码解析与操作逻辑解耦。

访问者模式的缺点

  1. 增加新元素类型困难
    • 缺点:每增加一种新元素类型,所有访问者类都需要修改。
    • 示例:如果在类结构中新增一种元素(如注解),所有访问者类都需要添加对应的visit方法。
  2. 破坏封装
    • 缺点:访问者可能需要访问元素的私有成员,从而破坏封装性。
    • 示例:访问者可能需要直接访问类的私有字段来完成某些操作。
  3. 复杂性增加
    • 缺点:访问者模式引入了额外的类和接口,增加了代码的复杂性。
    • 示例:需要定义Visitor接口、ConcreteVisitor 实现类以及Element 接口。
  4. 性能开销
    • 缺点:双重分派机制可能带来一定的性能开销。
    • 示例:在性能敏感的场景中,访问者模式可能不如直接操作对象结构高效。

访问者模式的适用场景

  1. 对象结构稳定,但操作频繁变化
    • 场景:对象结构很少变化,但需要频繁添加新的操作。
    • 示例:编译器中的语法树遍历(如代码优化、静态分析)。
  2. 需要对对象结构进行多种不相关的操作
    • 场景:对象结构需要支持多种不相关的操作,且希望将这些操作集中管理。
    • 示例:文档对象模型(DOM)的遍历和操作(如渲染、导出、校验)。
  3. 操作需要跨多个元素协作
    • 场景:某些操作需要访问多个元素并协作完成。
    • 示例:统计文档中所有图片的尺寸总和。
  4. 避免污染元素类的代码
    • 场景:希望保持元素类的简洁,避免将操作逻辑分散在各个元素类中。
    • 示例:ASM 框架通过访问者模式避免将字节码操作逻辑分散在类、方法、字段等元素中。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/19725.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

建筑兔零基础自学python记录22|实战人脸识别项目——视频人脸识别(下)11

这次我们继续解读代码&#xff0c;我们主要来看下面两个部分&#xff1b; 至于人脸识别成功的要点我们在最后总结~ 具体代码学习&#xff1a; #定义人脸名称 def name():#预学习照片存放位置path M:/python/workspace/PythonProject/face/imagePaths[os.path.join(path,f) f…

源代码防泄密沙箱是啥意思?

SDC沙盒通过多种技术手段实现环境隔离&#xff0c;从而有效防止数据泄露。以下是其具体的隔离机制&#xff1a; 1. 创建隔离的加密沙盒 SDC沙盒在员工电脑上虚拟出一个对外隔绝的加密沙盒。这个沙盒会主动与服务器进行认证对接&#xff0c;形成服务器-客户端沙盒这样一个涉密…

【复现DeepSeek-R1之Open R1实战】系列4:SFT和GRPO源码逐行深度解析(上)(3万字长文,从零开始到入门,包含详细的拓展基础知识)

目录 1 前言1.1 Open R1项目简介1.2 主要步骤1.3 原理图 2 基础知识2.1 Vocabulary和Tokenizer2.1.1 vocab.json, tokenizer.json, tokenizer_config.json2.1.2 什么是tokenizer2.1.3 在哪一步将tokenizer转成embedding2.1.4 tokenizer的代码实现 2.2 SFT和GRPO2.2.1 SFT2.2.2 …

课题推荐:高空长航无人机多源信息高精度融合导航技术研究

高空长航无人机多源信息高精度融合导航技术的研究&#xff0c;具有重要的理论意义与应用价值。通过深入研究多源信息融合技术&#xff0c;可以有效提升无人机在高空复杂环境下的导航能力&#xff0c;为无人机的广泛应用提供强有力的技术支持。希望该课题能够得到重视和支持&…

python_excel批量插入图片

提取excel的指定列的值的后4位&#xff08;数值&#xff09;&#xff0c;在其它列名的单元格中&#xff0c;批量嵌入与该数值匹配的图片&#xff08;未实现居中&#xff09;&#xff0c;每间隔4行处理一次&#xff08;合并过单元格&#xff09;。 import pandas as pd from ope…

DeepSeek 助力 Vue 开发:打造丝滑的颜色选择器(Color Picker)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

Jenkinsdebug:遇到ERROR: unable to select packages:怎么处理

报错信息&#xff1a; 报错信息解释&#xff1a; musl-1.2.5-r0 和 musl-dev-1.2.5-r1: 这里说明 musl-dev 需要一个特定版本的 musl&#xff0c;即 musl1.2.5-r1&#xff0c;但是当前版本的 musl&#xff08;1.2.5-r0&#xff09;并不满足这个条件。版本冲突: 当尝试安装新…

并查集基础+优化(下标从0开始)

#include<iostream> #include<algorithm> #include<vector> using namespace std; const int N 1e510; int n,m; int fa[N]; void set(int u,int v) {fa[v] u; } int find(int arr[],int i) {while(arr[i] ! -1){i arr[i]; } return i;//返回的是这个节点…

STM32创建静态库lib

创建静态库lib 1. 新建工程1.1 创建工程文件夹1.2 编写用户相关代码1.2.1 stm32f4xx_it.h1.2.2 stm32f4xx_it.c1.2.3 标准库配置&#xff1a;stm32f4xx_conf.h1.2.4 HAL库的配置&#xff1a;stm32f4xx_hal_conf.h1.2.5 LL库配置&#xff1a;stm32f4xx_ll_conf.h 1.3 移植通用文…

CV -- 基于GPU版显卡CUDA环境+Pycharm YOLOv8 检测

目录 下载 CUDA 下载 cuDNN 下载 anaconda 安装 PyTorch pycharm 搭配 yolo 环境并运行 阅读本文须知&#xff0c;需要电脑中有 Nvidia 显卡 下载 CUDA 打开 cmd &#xff0c;输入 nvidia-smi &#xff0c;查看电脑支持 CUDA 版本&#xff1a; 我这里是12.0&#xff0c;进入…

MATLAB图像处理:图像分割方法

图像分割将图像划分为具有特定意义的子区域&#xff0c;是目标检测、医学影像分析、自动驾驶等领域的核心预处理步骤。本文讲解阈值分割、边缘检测、区域生长、聚类分割、基于图的方法等经典与前沿技术&#xff0c;提供MATLAB代码实现。 目录 1. 图像分割基础 2. 经典分割方…

海康摄像头IPV6模式,手动,自动,路由公告

海康摄像头DS-2DC7220IW-A 网络设置中的IPv6配置选项。IPv6是互联网协议&#xff08;IP&#xff09;的第六版&#xff0c;用于替代IPv4&#xff0c;提供更多的IP地址和改进的网络功能。图片中的选项允许用户选择如何配置设备的IPv6网络连接&#xff1a; 手动&#xff1a;用户可…

NewMap10.3土地勘测定界自动化系统

“NewMap报件通”适用于建设项目用地土地勘测定界工作&#xff0c;其设计理念是以最大化提高作业效率与最简化作业员操作为原则&#xff0c;后台采用数据库管理技术&#xff0c;以“GIS概念”实现了图形数据与属性数据的双向联动&#xff0c;利用该系统可以方便快捷地绘制数字化…

栈(典型算法思想)—— OJ例题算法解析思路

目录 一、1047. 删除字符串中的所有相邻重复项 - 力扣&#xff08;LeetCode&#xff09; 算法代码&#xff1a; 1. 初始化结果字符串 2. 遍历输入字符串 3. 检查和处理字符 4. 返回结果 总结 二、844. 比较含退格的字符串 - 力扣&#xff08;LeetCode&#xff09; 算…

Qt中基于开源库QRencode生成二维码(附工程源码链接)

目录 1.QRencode简介 2.编译qrencode 3.在Qt中直接使用QRencode源码 3.1.添加源码 3.2.用字符串生成二维码 3.3.用二进制数据生成二维码 3.4.界面设计 3.5.效果展示 4.注意事项 5.源码下载 1.QRencode简介 QRencode是一个开源的库&#xff0c;专门用于生成二维码&…

字符串哈希动态规划_6

一.字符串哈希 字符串哈希概述 字符串哈希是一种将字符串映射到一个数值的技术&#xff0c;常用于处理字符串相关的算法问题&#xff0c;尤其在处理字符串匹配、子串查找等问题时非常高效。它的核心思想是利用一个哈希函数将字符串映射成一个整数&#xff0c;并根据该整数来判…

Kubernetes控制平面组件:Kubernetes如何使用etcd

云原生学习路线导航页&#xff08;持续更新中&#xff09; kubernetes学习系列快捷链接 Kubernetes架构原则和对象设计&#xff08;一&#xff09;Kubernetes架构原则和对象设计&#xff08;二&#xff09;Kubernetes架构原则和对象设计&#xff08;三&#xff09;Kubernetes控…

Mybatis后端数据库查询多对多查询解决方案

问题场景&#xff1a; 我开发的是一个论文选择系统。 后端用一个论文表paper来存储论文信息。 论文信息中&#xff0c;包含前置课程&#xff0c;也就是你需要修过这些课程才能选择这个论文。 而一个论文对应的课程有很多个。 这样就造成了一个数据库存储的问题。一个paper…

BGP配置华为——RR反射器配置

实验拓扑 与之前实验同理将loop0作为routerID使用&#xff0c;且R1和R2上用loop1接口用于模拟用户其他网段 实验要求 1&#xff0c;在AS100内运行OSPF协议 2.配置路由反射器&#xff0c;使得从R1进入的数据能够反射到全局网络 3.在R1和R2上分别宣告自己的loop1口网段用于观…

CentOS7 离线安装 Postgresql 指南

一、背景 服务器通常都是离线内网环境&#xff0c;想要通过联网方式一键下载安装 Postgresql 不太现实&#xff0c;本文将介绍如何在 CentOS7 离线安装 Postgresql&#xff0c;以及遇到困难如何解决。 二、安装包下载 先在本地下载好 rpm 包&#xff0c;再通过 ftp 上传到服…