java设计模式六 访问者
访问者模式(Visitor Pattern)是一种设计模式,它允许你将算法附加到对象结构中的各个元素上,而不必修改对象结构本身。它主要用于处理对象结构非常稳定,但频繁需要在此结构上执行不同操作的场景。访问者模式通过将操作移动到一个访问者对象中,来保持元素对象的类结构不变。
角色
- 元素(Element):定义一个接受访问者对象的接口。
- 具体元素(Concrete Element):实现元素接口,存储该元素的信息。
- 访问者(Visitor):定义一个访问元素的接口,每个元素类都有一个对应的访问方法。
- 具体访问者(Concrete Visitor):实现访问者接口,存储访问元素时的状态,并定义元素访问的操作。
案例分析
假设我们有一个表示文档的结构,文档中包含不同类型的内容,比如文本段落和图片。我们希望对这些内容执行不同的操作,比如计算总字数或者显示内容,而不修改现有的文档结构。
Java实现
首先,我们定义元素接口和具体元素:
// 元素接口
public interface DocumentElement {void accept(Visitor visitor);
}// 具体元素:文本段落
public class TextElement implements DocumentElement {private String content;public TextElement(String content) {this.content = content;}@Overridepublic void accept(Visitor visitor) {visitor.visit(this);}public String getContent() {return content;}
}// 具体元素:图片
public class ImageElement implements DocumentElement {private String url;public ImageElement(String url) {this.url = url;}@Overridepublic void accept(Visitor visitor) {visitor.visit(this);}public String getUrl() {return url;}
}
然后,我们定义访问者接口和具体访问者:
// 访问者接口
public interface Visitor {void visit(TextElement textElement);void visit(ImageElement imageElement);
}// 具体访问者:计算总字数
public class WordCountVisitor implements Visitor {private int wordCount;public WordCountVisitor() {wordCount = 0;}@Overridepublic void visit(TextElement textElement) {String[] words = textElement.getContent().split("\\s+");wordCount += words.length;}@Overridepublic void visit(ImageElement imageElement) {// 图片不计字数}public int getWordCount() {return wordCount;}
}// 具体访问者:显示内容
public class DisplayVisitor implements Visitor {@Overridepublic void visit(TextElement textElement) {System.out.println("Text: " + textElement.getContent());}@Overridepublic void visit(ImageElement imageElement) {System.out.println("Image: " + imageElement.getUrl());}
}
最后,我们使用访问者模式:
public class Document {private List<DocumentElement> elements = new ArrayList<>();public void addElement(DocumentElement element) {elements.add(element);}public void displayAll() {for (DocumentElement element : elements) {element.accept(new DisplayVisitor());}}public int getWordCount() {return new WordCountVisitor().visitAll(elements);}private int visitAll(List<DocumentElement> elements, Visitor visitor) {for (DocumentElement element : elements) {element.accept(visitor);}return 0; // 实际上,我们需要一个方法来累加计算结果}public static void main(String[] args) {Document document = new Document();document.addElement(new TextElement("Hello, World!"));document.addElement(new ImageElement("http://example.com/image.png"));document.addElement(new TextElement("This is a test document."));System.out.println("Displaying all elements:");document.displayAll();System.out.println("Total word count: " + document.getWordCount());}
}
注意事项
- 访问者模式使得增加新元素很容易,但增加新操作却需要修改所有的元素类,因此需要谨慎使用。
- 访问者模式破坏了封装,因为元素需要接受访问者,暴露内部信息。
- 访问者模式通常用于那些对象结构相对稳定,但经常需要在此结构上执行各种不同操作的场景。
通过上述案例分析,我们可以看到访问者模式如何在不改变现有对象结构的情况下,向对象添加新的操作。这种模式在某些特定场景下非常有用,但也需要考虑到其对封装性和扩展性的影响。
在上一个回答中,我们已经实现了访问者模式的一个基本例子,包括定义元素和访问者接口、具体实现以及如何在文档结构中应用访问者模式。接下来,我将提供一些额外的思考点和可能的改进方向。
改进方向
-
增加灵活性:考虑将访问者模式与命令模式结合使用,以提供更多的灵活性和可扩展性。
-
异步处理:如果访问操作非常耗时,可以考虑实现异步处理。
-
性能优化:在某些情况下,访问者模式可能会对性能产生影响,特别是当文档结构非常庞大时。需要考虑如何优化性能。
-
错误处理:在访问者模式中,错误处理非常重要。需要考虑如何处理访问过程中可能出现的异常。
-
访问者模式的变体:根据具体需求,可以考虑使用访问者模式的变体,如对称访问者模式或反向访问者模式。
访问者模式的变体
-
对称访问者模式:在这种变体中,访问者和元素是对称的,即访问者和元素都实现了相同的接口。这可以简化代码,但也牺牲了一些灵活性。
-
反向访问者模式:在这种变体中,元素对象访问访问者对象,而不是访问者访问元素。这在某些情况下可能更自然,但需要重新设计对象结构。
改进后的访问者模式实现
让我们对访问者模式实现进行一些改进,使其更加灵活和健壮:
public interface Visitor {void visitConcreteElementA(ConcreteElementA element);void visitConcreteElementB(ConcreteElementB element);// 可以添加更多的visit方法
}public class ConcreteElementA implements Element {@Overridepublic void accept(Visitor visitor) {visitor.visitConcreteElementA(this);}// 其他具体元素的方法
}public class ConcreteElementB implements Element {@Overridepublic void accept(Visitor visitor) {visitor.visitConcreteElementB(this);}// 其他具体元素的方法
}public class ConcreteVisitor1 implements Visitor {@Overridepublic void visitConcreteElementA(ConcreteElementA element) {// 处理ConcreteElementA}@Overridepublic void visitConcreteElementB(ConcreteElementB element) {// 处理ConcreteElementB}// 可以添加更多的visit方法
}public class ConcreteVisitor2 implements Visitor {@Overridepublic void visitConcreteElementA(ConcreteElementA element) {// 另一种处理ConcreteElementA的方式}@Overridepublic void visitConcreteElementB(ConcreteElementB element) {// 另一种处理ConcreteElementB的方式}// 可以添加更多的visit方法
}public class ObjectStructure {private List<Element> elements = new ArrayList<>();public void addElement(Element element) {elements.add(element);}public void operate(Visitor visitor) {for (Element element : elements) {element.accept(visitor);}}
}
使用改进后的访问者模式
public class Client {public static void main(String[] args) {ObjectStructure objectStructure = new ObjectStructure();objectStructure.addElement(new ConcreteElementA());objectStructure.addElement(new ConcreteElementB());System.out.println("Operation 1:");objectStructure.operate(new ConcreteVisitor1());System.out.println("Operation 2:");objectStructure.operate(new ConcreteVisitor2());}
}
在这个改进的例子中,我们为访问者模式增加了更多的灵活性,允许根据不同的访问者实现不同的操作。同时,我们也提供了更多的扩展点,以便于在不修改现有结构的情况下添加新的操作。
通过这些改进,访问者模式变得更加健壮和易于使用。它不仅提供了更高的灵活性,还提高了代码的可维护性和可读性。在实际应用中,根据具体需求,还可以继续对访问者模式进行定制和优化。
相关文章:
java设计模式六 访问者
访问者模式(Visitor Pattern)是一种设计模式,它允许你将算法附加到对象结构中的各个元素上,而不必修改对象结构本身。它主要用于处理对象结构非常稳定,但频繁需要在此结构上执行不同操作的场景。访问者模式通过将操作移…...
中间件研发之Springboot自定义starter
Spring Boot Starter是一种简化Spring Boot应用开发的机制,它可以通过引入一些预定义的依赖和配置,让我们快速地集成某些功能模块,而无需繁琐地编写代码和配置文件。Spring Boot官方提供了很多常用的Starter,例如spring-boot-star…...
libcity笔记:添加新模型(以RNN.py为例)
创建的新模型应该继承AbstractModel或AbstractTrafficStateModel 交通状态预测任务——>继承 AbstractTrafficStateModel类轨迹位置预测任务——>继承AbstractModel类 1 AbstractTrafficStateModel 2 RNN 2.1 构造函数 2.2 predict 2.3 calculate_loss...
Ansible---自动化运维工具
一、Ansible概述 1.1 Ansible简介 Ansible是一款自动化运维工具,通过ssh对目标主机进行配置、应用部署、任务执行、编排调度等操作。它简化了复杂的环境管理和自动化任务,提高了工作效率和一致性,同时,Ansible的剧本(playbooks)…...
5.Git
Git是一个分布式版本控制工具,主要用于管理开发过程中的源代码文件(Java类、xml文件、html文件等)。通过Git仓库来存储和管理这些文件,Git仓库分为两种 本地仓库:开发人员自己电脑上的Git仓库远程仓库:远程…...
探索中位数快速排序算法:高效寻找数据集的中间值
在计算机科学领域,寻找数据集的中位数是一个常见而重要的问题。而快速排序算法作为一种高效的排序算法,可以被巧妙地利用来解决中位数查找的问题。本文将深入探讨中位数快速排序算法的原理、实现方法以及应用场景,带你领略这一寻找中间值的高…...
密码学《图解密码技术》 记录学习 第十五章
目录 十五章 15.1本章学习的内容 15.2 密码技术小结 15.2.1 密码学家的工具箱 15.2.2 密码与认证 15.2.3 密码技术的框架化 15.2.4 密码技术与压缩技术 15.3 虚拟货币——比特币 15.3.1 什么是比特币 15.3.2 P2P 网络 15.3.3地址 15.3.4 钱包 15.3.5 区块链 15.3.…...
如何在 Ubuntu 16.04 上为 Nginx 创建自签名 SSL 证书
简介 TLS,即传输层安全协议,及其前身SSL,即安全套接字层,是用于将普通流量包装在受保护的加密包装中的网络协议。 使用这项技术,服务器可以在服务器和客户端之间安全地发送流量,而不会被外部方拦截。证书…...
5.协议的编解码
本章内容其实没有多大难度,主要考察大家的细心程度.计算数据长度然后截取相应字节数组并按照协议进行解码,编码则反之。 1.基础消息的编解码 Override public BasicMessage decode(byte[] bytes) {int dataLength ByteUtil.bytesToInt(ByteUtil.extra…...
数据结构基础| 线性表
线性表 定义 没有元素则为空表 例子: 稀疏多项式的运算 图书信息管理系统 特点 线性结构 同类型 线性表的类型定义 1.基本操作: InitList(&L) 操作结果:构造空的线性表L DestroyList(&L) 初始化条件:线性表L存在 操作结果:销毁线性表L(线性表L不存在) Cle…...
嵌入式学习
笔记 作业 有如下结构体 struct Student{ char name[16]; int age; double math_score; double chinese_score; double english_score; double physics_score; double chemistry…...
sass-loader和node-sass与node版本的依赖问题
sass-loader和node-sass与node版本的依赖问题 没有人会陪你走到最后,碰到了便是有缘,即使到了要下车的时候,也要心存感激地告别,在心里留下空白的一隅之地,多年后想起时依然心存甘味。——林清玄 报错截图 报错信息 np…...
基于BP神经网络的QPSK解调算法matlab性能仿真
目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 ........................................................................ for ij 1:leng…...
Linux服务器常用巡检命令
在Linux服务器上进行常规巡检是确保服务器稳定性和安全性的重要措施之一。以下是一些常用的巡检命令和技巧: 1. 查看系统信息 1.1 系统信息显示 命令:uname -a [rootlinux100 ~]# uname -a Linux linux100 4.15.0-70-generic #79-Ubuntu SMP…...
VSCode 配置 CMake
VSCode 配置 C/C 环境的详细过程可参考:VSCode 配置 C/C 环境 1 配置C/C编译环境 如果是 Windows 环境,需要安装 MingW。 方案一 可以去官网(https://sourceforge.net/projects/mingw-w64/)下载安装包。 注意安装路径不要出现中文。 打开 windows she…...
《MATLAB科研绘图与学术图表绘制从入门到精通》示例:绘制德国每日风能和太阳能产量3D线图
在MATLAB中,要绘制3D线图,可以使用 plot3 函数。 在《MATLAB科研绘图与学术图表绘制从入门到精通》书中通过绘制德国每日风能和太阳能产量3D线图解释了如何在MATLAB中绘制3D线图。 购书地址:https://item.jd.com/14102657.html...
【信息系统项目管理师知识点速记】质量管理:控制质量
控制质量是为了评估绩效,确保项目输出完整、正确且满足客户期望,而监督和记录质量管理活动执行结果的过程。控制质量过程需要在整个项目期间开展,其目的是测量产品或服务的完整性、合规性和适用性,以确保项目达到主要干系人的质量要求。 12.5.1 输入 项目管理计划 质量管理…...
【云原生】Pod 的生命周期(一)
【云原生】Pod 的生命周期(一)【云原生】Pod 的生命周期(二) Pod 的生命周期(一) 1.Pod 生命期2.Pod 阶段3.容器状态3.1 Waiting (等待)3.2 Running(运行中)3…...
Golang | Leetcode Golang题解之第71题简化路径
题目: 题解: func simplifyPath(path string) string {stack : []string{}for _, name : range strings.Split(path, "/") {if name ".." {if len(stack) > 0 {stack stack[:len(stack)-1]}} else if name ! "" &am…...
Unreal游戏GPU性能优化检测模式全新上线
UWA已经在去年推出了针对于Unity项目的GPU性能优化工具,通过对GPU渲染性能、带宽性能以及各种下探指标,帮助Unity项目研发团队定位由GPU导致的发热耗电问题。这个需求在Unreal团队中也极为强烈,因此UWA将该功能移植到针对Unreal项目的GOT Onl…...
Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...
成都鼎讯硬核科技!雷达目标与干扰模拟器,以卓越性能制胜电磁频谱战
在现代战争中,电磁频谱已成为继陆、海、空、天之后的 “第五维战场”,雷达作为电磁频谱领域的关键装备,其干扰与抗干扰能力的较量,直接影响着战争的胜负走向。由成都鼎讯科技匠心打造的雷达目标与干扰模拟器,凭借数字射…...
视频行为标注工具BehaviLabel(源码+使用介绍+Windows.Exe版本)
前言: 最近在做行为检测相关的模型,用的是时空图卷积网络(STGCN),但原有kinetic-400数据集数据质量较低,需要进行细粒度的标注,同时粗略搜了下已有开源工具基本都集中于图像分割这块,…...
安卓基础(Java 和 Gradle 版本)
1. 设置项目的 JDK 版本 方法1:通过 Project Structure File → Project Structure... (或按 CtrlAltShiftS) 左侧选择 SDK Location 在 Gradle Settings 部分,设置 Gradle JDK 方法2:通过 Settings File → Settings... (或 CtrlAltS)…...
Python网页自动化Selenium中文文档
1. 安装 1.1. 安装 Selenium Python bindings 提供了一个简单的API,让你使用Selenium WebDriver来编写功能/校验测试。 通过Selenium Python的API,你可以非常直观的使用Selenium WebDriver的所有功能。 Selenium Python bindings 使用非常简洁方便的A…...
spring Security对RBAC及其ABAC的支持使用
RBAC (基于角色的访问控制) RBAC (Role-Based Access Control) 是 Spring Security 中最常用的权限模型,它将权限分配给角色,再将角色分配给用户。 RBAC 核心实现 1. 数据库设计 users roles permissions ------- ------…...
文件上传漏洞防御全攻略
要全面防范文件上传漏洞,需构建多层防御体系,结合技术验证、存储隔离与权限控制: 🔒 一、基础防护层 前端校验(仅辅助) 通过JavaScript限制文件后缀名(白名单)和大小,提…...
Spring Boot 中实现 HTTPS 加密通信及常见问题排查指南
Spring Boot 中实现 HTTPS 加密通信及常见问题排查指南 在金融行业安全审计中,未启用HTTPS的Web应用被列为高危漏洞。通过正确配置HTTPS,可将中间人攻击风险降低98%——本文将全面解析Spring Boot中HTTPS的实现方案与实战避坑指南。 一、HTTPS 核心原理与…...
Life:Internship finding
1. 前言 fishwheel writes this Blog to 记录自分自身在研二下找实习的经历。When 写这篇 Blog 的时候我的最后一搏也挂掉了,只能启用保底方案了。When I 打开我的邮箱时,发现里面有 nearly 100 多封与之相关的邮件,顿时感到有些心凉&#x…...
174页PPT家居制造业集团战略规划和运营管控规划方案
甲方集团需要制定一个清晰的集团价值定位,从“指引多元”、“塑造 能力”以及“强化协同”等方面引领甲方做大做强 集团需要通过管控模式、组织架构及职能、授权界面、关键流程、战略 实施和组织演进路径,平衡风险控制和迅速发展,保证战略落地…...
