访问者模式的理解和实践
在软件开发过程中,设计模式为我们提供了解决常见问题的最佳实践。访问者模式(Visitor Pattern)是行为设计模式之一,它将数据操作与数据结构分离,使得在不修改数据结构的前提下,能够定义作用于这些元素的新的操作。本文将详细讲解访问者模式的概念、原理、优缺点,并通过Java代码示例展示其在实际项目中的应用。
一、访问者模式的概念
访问者模式是一种将数据操作与数据结构分离的设计模式。它通过将作用于某种数据结构中的各元素的操作封装起来,使得这些操作可以独立于数据结构进行变化。访问者模式使得我们能够在不修改数据结构的前提下,增加新的操作。
二、访问者模式的结构
访问者模式包含以下几个角色:
- Visitor(访问者):接口或抽象类,声明了访问者对各个元素的操作方法。
- ConcreteVisitor(具体访问者):实现了Visitor接口或抽象类,具体实现了对各个元素的操作。
- Element(元素):接口或抽象类,声明了接受访问者的方法。
- ConcreteElement(具体元素):实现了Element接口或抽象类,存储数据,并实现了接受访问者的方法。
- ObjectStructure(对象结构):包含多个元素,可以迭代这些元素,并允许访问者访问这些元素。
三、访问者模式的原理
访问者模式的原理是将操作从数据结构中分离出来,封装到访问者类中。数据结构中的每个元素都接受访问者对象,访问者对象通过访问这些元素来执行相应的操作。这样,当需要增加新的操作时,只需新增一个访问者类,而无需修改数据结构。
四、访问者模式的优缺点
优点:
- 增加新的操作很容易:只需增加一个新的访问者类,而无需修改已有的数据结构。
- 将数据操作集中管理:访问者模式将相关的操作集中到一个访问者类中,便于管理。
- 分离了数据结构和操作:数据结构和操作不再耦合在一起,提高了系统的灵活性。
缺点:
- 增加了类的数量:每增加一个新的操作,都需要增加一个新的访问者类,增加了类的数量。
- 破坏了封装:访问者需要访问被访问对象的内部结构,这在一定程度上破坏了封装性。
- 增加了系统复杂度:访问者模式的实现相对复杂,需要理解其工作原理,才能正确使用。
五、访问者模式的实践
下面通过Java代码示例,展示访问者模式在实际项目中的应用。
示例背景:
假设我们有一个简单的员工管理系统,员工分为两类:工程师(Engineer)和经理(Manager)。我们需要实现两个操作:计算工资(CalculateSalary)和显示员工信息(DisplayInfo)。
代码实现:
定义Element接口:
// 定义Element接口
public interface Element {void accept(Visitor visitor);
}
定义具体元素类:
// 定义工程师类
public class Engineer implements Element {private String name;private int salary;public Engineer(String name, int salary) {this.name = name;this.salary = salary;}public String getName() {return name;}public int getSalary() {return salary;}@Overridepublic void accept(Visitor visitor) {visitor.visit(this);}
}// 定义经理类
public class Manager implements Element {private String name;private int salary;private int bonus;public Manager(String name, int salary, int bonus) {this.name = name;this.salary = salary;this.bonus = bonus;}public String getName() {return name;}public int getSalary() {return salary;}public int getBonus() {return bonus;}@Overridepublic void accept(Visitor visitor) {visitor.visit(this);}
}
定义Visitor接口:
// 定义Visitor接口
public interface Visitor {void visit(Engineer engineer);void visit(Manager manager);
}
定义具体访问者类:
// 定义计算工资访问者类
public class CalculateSalaryVisitor implements Visitor {@Overridepublic void visit(Engineer engineer) {System.out.println("Engineer " + engineer.getName() + " salary: " + engineer.getSalary());}@Overridepublic void visit(Manager manager) {int totalSalary = manager.getSalary() + manager.getBonus();System.out.println("Manager " + manager.getName() + " salary: " + totalSalary);}
}// 定义显示信息访问者类
public class DisplayInfoVisitor implements Visitor {@Overridepublic void visit(Engineer engineer) {System.out.println("Engineer: " + engineer.getName());}@Overridepublic void visit(Manager manager) {System.out.println("Manager: " + manager.getName() + ", Bonus: " + manager.getBonus());}
}
定义ObjectStructure类:
import java.util.ArrayList;
import java.util.List;// 定义ObjectStructure类
public class ObjectStructure {private List<Element> elements = new ArrayList<>();public void addElement(Element element) {elements.add(element);}public void removeElement(Element element) {elements.remove(element);}public void accept(Visitor visitor) {for (Element element : elements) {element.accept(visitor);}}
}
客户端代码:
public class Client {public static void main(String[] args) {ObjectStructure os = new ObjectStructure();os.addElement(new Engineer("John Doe", 70000));os.addElement(new Manager("Jane Smith", 80000, 10000));Visitor calculateSalaryVisitor = new CalculateSalaryVisitor();os.accept(calculateSalaryVisitor);System.out.println("------");Visitor displayInfoVisitor = new DisplayInfoVisitor();os.accept(displayInfoVisitor);}
}
运行结果:
Engineer John Doe salary: 70000
Manager Jane Smith salary: 90000
------
Engineer: John Doe
Manager: Jane Smith, Bonus: 10000
总结
访问者模式通过将操作从数据结构中分离出来,提高了系统的灵活性和可扩展性。它使得在不修改数据结构的前提下,能够增加新的操作。然而,访问者模式也增加了类的数量,破坏了封装,增加了系统的复杂度。因此,在实际应用中,我们需要根据具体需求权衡利弊,选择是否使用访问者模式。
通过上面的示例,我们可以看到访问者模式在员工管理系统中的应用,通过定义不同的访问者类,实现了计算工资和显示员工信息的功能。这使得系统的操作更加灵活,易于扩展和维护。希望这篇文章能够帮助大家更好地理解访问者模式,并在实际项目中灵活运用。
相关文章:
访问者模式的理解和实践
在软件开发过程中,设计模式为我们提供了解决常见问题的最佳实践。访问者模式(Visitor Pattern)是行为设计模式之一,它将数据操作与数据结构分离,使得在不修改数据结构的前提下,能够定义作用于这些元素的新的…...
在Scala中对Map函数的使用
package pp28{object cscc {def main(args: Array[String]): Unit {val m1 Map("马云 — 阿里巴巴" -> 1964,"马化腾 — 腾讯" -> 1971,"李彦宏 - 百度" -> 1968,"雷军 - 小米" -> 1969,"丁磊 - 网易" -> …...
PyTorch基本使用-张量的索引操作
在操作张量时,经常要去获取某些元素进行处理或者修改操作,在这里需要了解torch中的索引操作。 准备数据: data torch.randint(0,10,[4,5]) print(data--->,data)输出结果: data---> tensor([[3, 9, 4, 0, 5],[7, 5, 9, …...
OpenCV实验:图片加水印
第二篇:图片添加水印(加 logo) 1. 实验原理 水印原理: 图片添加水印是图像叠加的一种应用,分为透明水印和不透明水印。水印的实现通常依赖于像素值操作,将水印图片融合到目标图片中,常用的方法…...
sql server log文件
确定 SQL Server 实例中具有大量 VDF 的数据库 SELECT [name], COUNT(l.database_id) AS vlf_count FROM sys.databases AS s CROSS APPLY sys.dm_db_log_info(s.database_id) AS l GROUP BY [name] HAVING COUNT(l.database_id) > 100; 在收缩日志文件之前确定事务日志中…...
Elasticsearch 集群部署
Elasticsearch 是一个分布式的搜索和分析引擎,广泛应用于日志分析、全文搜索、实时数据分析等场景。它以其高性能、高可用性和易用性而著称。本文档将引导您完成一个基本的 Elasticsearch 集群配置,包括节点间的通信、客户端访问、安全设置等关键步骤。我…...
微信小程序5-图片实现点击动作和动态加载同类数据
搜索 微信小程序 “动物觅踪” 观看效果 感谢阅读,初学小白,有错指正。 一、功能描述 a. 原本想通过按钮加载背景图片,来实现一个可以点击的搜索button,但是遇到两个难点,一是按钮大小调整不方便(网上搜索…...
策略梯度定理公式的详细推导
策略梯度定理公式的详细推导 以下是策略梯度定理公式从基础概率公式到最终形式的完整推导,帮助更清晰地理解推导过程中的每一个步骤。 1. 策略梯度的目标 我们希望最大化期望累积奖励 ( J ( θ ) J(\theta) J(θ) ),其定义为: J ( θ ) E…...
力扣-图论-10【算法学习day.60】
前言 ###我做这类文章一个重要的目的还是给正在学习的大家提供方向和记录学习过程(例如想要掌握基础用法,该刷哪些题?)我的解析也不会做的非常详细,只会提供思路和一些关键点,力扣上的大佬们的题解质量是非…...
《Python WEB安全 库全解析》
《Python WEB安全 库全解析》 一、Python WEB安全 库概述二、常见的 Python WEB安全 库介绍1. Jiasule2. Awesome Python Security3. Flask-Security4. Flask-SeaSurf 三、Python WEB 安全库的优缺点1. 优点2. 缺点 四、Python WEB 安全库的使用场景1. 开发 Web 应用2. 处理敏感…...
Linux yum-config-manager命令异常
错误信息 使用 yum-config-manager命令时错误信息如下 sudo yum-config-manager \ > --add-repo \ > https://download.docker.com/linux/centos/docker-ce.repo sudo: yum-config-manager: command not found 解决办法 第一步: sudo yum -y install yum-u…...
ios 开发配置蓝牙
如果使用了蓝牙功能, 又没有配置, 会出现以下错误: This app has crashed because it attempted to access privacy-sensitive data without a usage description. The apps Info.plist must contain an NSBluetoothAlwaysUsageDescription key with a string value explaini…...
geoserver(1) 发布sql 图层 支持自定义参数
前提使用postgis 数据库支持关联 join 支持 in,not in,like,及其他sql原生函数 新增sql图层 编写自定义sql 编辑sql语句必须输出带有geom数据 正则表达式去除 设置id以及坐标参考系 预览sql图层效果 拼接sql参数 http://xxx.com/geoserver/weather/wms?SERVICEWMS&VERSI…...
Linux:network:添加ip的时候自动添加一个本地路由
文章目录 问题问题 最近在看一个路由的问题,顺便看内核代码,发现在添加IP的时候,内核会自动添加一个local route。 net/ipv4/devinet.c inet_rtm_newaddr->__inet_insert_ifa /* Send message first, then call notifier.Notifier will trigger FIB update, so thatlis…...
go 集成nacos注册中心、配置中心
使用限制 Go>v1.15 Nacos>2.x 安装 使用go get安装SDK: go get -u github.com/nacos-group/nacos-sdk-go/v2 快速使用 初始化客户端配置ClientConfig constant.ClientConfig{TimeoutMs uint64 // 请求Nacos服务端的超时时间,默…...
ssd202d-badblock-坏块检测
这边文章讲述的是坏快检测功能 思路: 1.第一次烧录固件会实现跳坏块,但是后续使用会导致坏块的产生; 于是我在uboot环境变量添加了两个变量来控制坏快 lb_badnum //坏块个数 lb_badoff //坏块所在位置 2.第一次开机会根据lb_badnum是否…...
MySQL-练习-数据介绍
文章目录 一. 数据介绍1. 数据结构2. 创建数据库,数据表3. 员工表(employees)练习1 4. 顾客表(customers)练习2 5. 商品(products)和商品类别(categories)表练习3 6. 供应商表(suppliers)练习4 7. 订单和订单明细表练习5 二. 数据汇总三. 使用CASE WHEN …...
React框架:解锁现代化Web开发的新维度
在当今前端开发领域,React 无疑是一颗璀璨的明星。React 是由 Facebook 开发的用于构建用户界面的 JavaScript 库,它在前端开发中占据着重要的地位,为开发者提供了一种高效、灵活且可维护的方式来构建复杂的用户界面。 一、React 的背景与开…...
电阻功率,限流,等效电阻
1 电阻额定功率 2 电阻限流作用 3 电阻并联等效电阻...
Qt | 开发工具(top1)
Qt Creator 跨平台、完整的集成开发环境(IDE),供应用程序开发者创建用于多个桌面、嵌入式和移动设备平台的应用程序。 Qt Linguist 一套将Qt C和Qt Quick应用程序翻译成本地语言的工具。 qmake Qt自动化构建工具,简化了不同平台的构建过程。…...
Node.js express
1. express 介绍 express 是一个基于 Node.js 平台的极简、灵活的 WEB 应用开发框架,官方网址:https://www.expressjs.com.cn/简单来说,express 是一个封装好的工具包,封装了很多功能,便于我们开发 WEB 应用ÿ…...
ios h5中在fixed元素中的input被focus时,键盘遮挡input (van-popup、van-feild)
问题描述: 前提:我使用的是vant组件库,其中一个页面中有一个van-popup组件,van-popup组件中又嵌套了一个van-field组件预期结果:当点击van-feild输入框时,键盘弹起,输入框显示在键盘上方实际结…...
springboot整合lua脚本在Redis实现商品库存扣减
1、目的 使用lua脚本,可以保证多条命令的操作原子性;同时可以减少操作IO(比如说判断redis对应数据是否小于0,小于0就重置为100,这个场景一般是取出来再判断,再存放进行,就至少存在2次IO,用lua脚…...
MySQL ON DUPLICATE KEY UPDATE影响行数
目录 分析为什么Updates返回7 总结 数据库更新日志如下 insertOrUpdateList|> Preparing: INSERT INTO clue_user_tag (vuid, tag_id, tag_type, content) VALUES (?, ?, ?, ?) , (?, ?, ?, ?) , (?, ?, ?, ?) , (?, ?, ?, ?) ON DUPLICATE KEY UPDATE …...
uniapp小程序 slot中无法传递外部参数的解决方案
最近在封装一个List组件,外部传给我数据,我循环后将每个Item部分slot到外部,由调用者自己去写item布局,类似ElementUI、iView的Tabe列表。 List: <view v-if"list.length > 0" class"list-scroll__item&quo…...
umi实现动态获取菜单权限
文章目录 前景登录组件编写登录逻辑菜单的时机动态路由页面刷新手动修改地址 前景 不同用户拥有不同的菜单权限,现在我们实现登录动态获取权限菜单。 登录组件编写 //当我们需要使用dva的dispatch函数时,除了通过connect函数包裹组件还可以使用这种方…...
Pytest-Bdd-Playwright 系列教程(14):Docstring 参数
Pytest-Bdd-Playwright 系列教程(14):Docstring 参数 前言一、什么是docstring?二、基本语法三、主要特点四、实际例子五、注意事项六、使用建议总结 前言 在自动化测试的过程中,我们经常需要处理复杂的测试数据或需要输入多行文…...
交互开发---测量工具(适用VTK或OpenGL开发的应用程序)
简介: 经常使用RadiAnt DICOM Viewer来查看DICOM数据,该软件中的测量工具比较好用,就想着仿照其交互方式自己实现下。后采用VTK开发应用程序时,经常需要开发各种各样的测量工具,如果沿用VTK的widgets的思路,…...
Qt 一个简单的QChart 绘图
Qt 一个简单的QChart 绘图 先上程序运行结果图: “sample9_1QChart.h” 文件代码如下: #pragma once#include <QtWidgets/QMainWindow> #include "ui_sample9_1QChart.h"#include <QtCharts> //必须这么设置 QT_CHARTS_USE_NAME…...
【Java笔记】LinkedList 底层结构
一、LinkedList 的全面说明 LinkedList底层实现了双向链表和双端队列特点可以添加任意元素(元素可以重复),包括null线程不安全,没有实现同步 二、LinkedList 的底层操作机制 三、LinkedList的增删改查案例 public class LinkedListCRUD { public stati…...
有没有接活做的网站/google chrome官网入口
初学者在学习向上转型可能会很难理解,向上转型并不能调用子类特有属性和方法,我们必须先生成子类实例再赋值给父类引用(向上转型),然后将父类引用向下强制转换给子类引用(向下转型),这样才能调用子类中的所有成员。这看起来像是多…...
福州自适应网站建设/海淀区seo全面优化
1.安装git2.如图所示:在AS 的File->Settings->Version Control->Git 配置git.exe命令路径,然后点击Test,提示successfully,则配置成功。3.在gitosc上创建仓库,获取仓库地址。4.创建AS项目。5.在AS中选择项目,…...
什么网站可以做数据调查/企业品牌推广方案
package code.test;import java.util.Comparator; import java.util.Iterator; import java.util.PriorityQueue; import java.util.Queue;/*** 实验表明,在java中:* 1.toString()方法或迭代元素:优先级队列打印或者迭代,得到的输…...
做自营网站还是amazon/温州seo团队
好了,就不多说了,事实上只要你有了足够的能力,能开始想老板所想的事,这时候你会发现,你也可以做老板了。 据我所知,很多程序员都是越过架构师这道坎,有了整体的架构布局思维,就开始…...
网站域名多少钱一年/交换友链平台
Cannot make a static reference to the non-static method的解决方法 标签: referencepropertiesstringexceptionthreadimport2012-09-13 12:01 45741人阅读 评论(2) 收藏 举报分类:JAVA(38) 版权声明:本文为博主原创…...
做一个模板网站多少钱/品牌传播策划方案
面试“秒杀”——开口决定胜负 会说话的人一般在很多事情中占了不少优势,同样的表达意思,有的人能把别人说笑了,有的人则能把别人说哭了。虽然,光说不练是“假把式”,但对于面试这种直接交流的形式,良好的口…...