当前位置: 首页 > news >正文

访问者模式的理解和实践

        在软件开发过程中,设计模式为我们提供了解决常见问题的最佳实践。访问者模式(Visitor Pattern)是行为设计模式之一,它将数据操作与数据结构分离,使得在不修改数据结构的前提下,能够定义作用于这些元素的新的操作。本文将详细讲解访问者模式的概念、原理、优缺点,并通过Java代码示例展示其在实际项目中的应用。

 

一、访问者模式的概念

        访问者模式是一种将数据操作与数据结构分离的设计模式。它通过将作用于某种数据结构中的各元素的操作封装起来,使得这些操作可以独立于数据结构进行变化。访问者模式使得我们能够在不修改数据结构的前提下,增加新的操作。

二、访问者模式的结构

        访问者模式包含以下几个角色:

  1. Visitor(访问者):接口或抽象类,声明了访问者对各个元素的操作方法。
  2. ConcreteVisitor(具体访问者):实现了Visitor接口或抽象类,具体实现了对各个元素的操作。
  3. Element(元素):接口或抽象类,声明了接受访问者的方法。
  4. ConcreteElement(具体元素):实现了Element接口或抽象类,存储数据,并实现了接受访问者的方法。
  5. ObjectStructure(对象结构):包含多个元素,可以迭代这些元素,并允许访问者访问这些元素。

三、访问者模式的原理

        访问者模式的原理是将操作从数据结构中分离出来,封装到访问者类中。数据结构中的每个元素都接受访问者对象,访问者对象通过访问这些元素来执行相应的操作。这样,当需要增加新的操作时,只需新增一个访问者类,而无需修改数据结构。

四、访问者模式的优缺点

优点

  1. 增加新的操作很容易:只需增加一个新的访问者类,而无需修改已有的数据结构。
  2. 将数据操作集中管理:访问者模式将相关的操作集中到一个访问者类中,便于管理。
  3. 分离了数据结构和操作:数据结构和操作不再耦合在一起,提高了系统的灵活性。

缺点

  1. 增加了类的数量:每增加一个新的操作,都需要增加一个新的访问者类,增加了类的数量。
  2. 破坏了封装:访问者需要访问被访问对象的内部结构,这在一定程度上破坏了封装性。
  3. 增加了系统复杂度:访问者模式的实现相对复杂,需要理解其工作原理,才能正确使用。

五、访问者模式的实践

        下面通过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


总结

        访问者模式通过将操作从数据结构中分离出来,提高了系统的灵活性和可扩展性。它使得在不修改数据结构的前提下,能够增加新的操作。然而,访问者模式也增加了类的数量,破坏了封装,增加了系统的复杂度。因此,在实际应用中,我们需要根据具体需求权衡利弊,选择是否使用访问者模式。

        通过上面的示例,我们可以看到访问者模式在员工管理系统中的应用,通过定义不同的访问者类,实现了计算工资和显示员工信息的功能。这使得系统的操作更加灵活,易于扩展和维护。希望这篇文章能够帮助大家更好地理解访问者模式,并在实际项目中灵活运用。

http://www.lryc.cn/news/502603.html

相关文章:

  • 在Scala中对Map函数的使用
  • PyTorch基本使用-张量的索引操作
  • OpenCV实验:图片加水印
  • sql server log文件
  • Elasticsearch 集群部署
  • 微信小程序5-图片实现点击动作和动态加载同类数据
  • 策略梯度定理公式的详细推导
  • 力扣-图论-10【算法学习day.60】
  • 《Python WEB安全 库全解析》
  • Linux yum-config-manager命令异常
  • ios 开发配置蓝牙
  • geoserver(1) 发布sql 图层 支持自定义参数
  • Linux:network:添加ip的时候自动添加一个本地路由
  • go 集成nacos注册中心、配置中心
  • ssd202d-badblock-坏块检测
  • MySQL-练习-数据介绍
  • React框架:解锁现代化Web开发的新维度
  • 电阻功率,限流,等效电阻
  • Qt | 开发工具(top1)
  • Node.js express
  • ios h5中在fixed元素中的input被focus时,键盘遮挡input (van-popup、van-feild)
  • springboot整合lua脚本在Redis实现商品库存扣减
  • MySQL ON DUPLICATE KEY UPDATE影响行数
  • uniapp小程序 slot中无法传递外部参数的解决方案
  • umi实现动态获取菜单权限
  • Pytest-Bdd-Playwright 系列教程(14):Docstring 参数
  • 交互开发---测量工具(适用VTK或OpenGL开发的应用程序)
  • Qt 一个简单的QChart 绘图
  • 【Java笔记】LinkedList 底层结构
  • el-table组件树形数据修改展开箭头