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

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

介绍

访问者模式(Visitor),表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变个元素的类的前提下定义作用于这些元素的新操作。

问题:在一个机构里面有两种员工,1.Teacher   2.Engineer   员工有name, income, vacationDays三种属性,想要对不同的员工有不同的算法实现更新income和vacationDays的操作。

首先引导出策略模式,对员工实现不同的算法策略实现更新操作。

基础代码

test类
package Stragy;import java.util.ArrayList;
import java.util.List;public class test {public static void main(String[] args) {Employee e1 = new Teacher("张三", 8000, 10);Employee e2 = new Engineer("李四", 7000, 20);Visitor visitor = new IncomeVisitor();List<Employee> employeeList = new ArrayList<>();employeeList.add(e1);employeeList.add(e2);for(Employee e : employeeList){visitor.visit(e);}for(Employee e : employeeList){System.out.println(e);}}
}
 Employee类
package Stragy;public abstract class Employee {private String name;private double income;private int vacationDays;public Employee(String name, double income, int vacationDays) {this.name = name;this.income = income;this.vacationDays = vacationDays;}public String getName() {return name;}public void setName(String name) {this.name = name;}public double getIncome() {return income;}public void setIncome(double income) {this.income = income;}public int getVacationDays() {return vacationDays;}public void setVacationDays(int vacationDays) {this.vacationDays = vacationDays;}@Overridepublic String toString() {return "Employee{" +"name='" + name + '\'' +", income=" + income +", vacationDays=" + vacationDays +'}';}
}
Engineer类
package Stragy;public class Engineer extends Employee{public Engineer(String name, double income, int vacationDays) {super(name, income, vacationDays);}}
 Teacher类
package Stragy;public class Teacher extends Employee {public Teacher(String name, double income, int vacationDays) {super(name, income, vacationDays);}
}
Visitor类 
package Stragy;public abstract class Visitor {public abstract void visit(Employee e);
}
IncomeVisitor类
package Stragy;public class IncomeVisitor extends Visitor{@Overridepublic void visit(Employee e) {if(e instanceof Teacher){e.setIncome(e.getIncome() + 500);}if(e instanceof Engineer){e.setIncome(e.getIncome() + 800);}}
}
 VacationDaysVisitor类
package Stragy;public class VacationDayVisitor extends Visitor{@Overridepublic void visit(Employee e) {e.setVacationDays(e.getVacationDays() + 8);}
}

策略模式

基础代码就是按照策略模式实现的,但是在不同的员工对算法的不同选择上尚可优化,在策略模式中,是通过 if 语句判断算法的不同选择。

package Stragy;public class IncomeVisitor extends Visitor{@Overridepublic void visit(Employee e) {if(e instanceof Teacher){e.setIncome(e.getIncome() + 500);}if(e instanceof Engineer){e.setIncome(e.getIncome() + 800);}}
}

在IncomeVisitor类中,通过instanceof判断当前员工是哪种类型来选择实现哪种算法,使用访问者的双分派方式可以省去 if 语句,也就是使用两次动态绑定。 

访问者模式

在访问者模式中,修改了以下内容:

在Employee类中添加accept抽象方法

public abstract void accept(Visitor visitor);

 在Engineer类中重写

@Overridepublic void accept(Visitor visitor) {visitor.EngineerVisit(this);}

在Teacher类中重写

@Overridepublic void accept(Visitor visitor) {visitor.TeacherVisit(this);}

在Visitor类中不再使用visit方法,替换为不同员工的算法选择,在此处,我的理解是用两个不同的方法代替了 if 语句。

package visitor;public abstract class Visitor {public abstract void TeacherVisit(Employee e);public abstract void EngineerVisit(Employee e);
}

在VacationDaysVisitor中重写

package visitor;public class VacationDaysVisitor extends Visitor{@Overridepublic void TeacherVisit(Employee e) {e.setVacationDays(e.getVacationDays() + 5);}@Overridepublic void EngineerVisit(Employee e) {e.setVacationDays(e.getVacationDays() + 8);}
}

在IncomeVisitor中重写

package visitor;public class IncomeVisitor extends Visitor{@Overridepublic void TeacherVisit(Employee e) {e.setIncome(e.getIncome() + 500);}@Overridepublic void EngineerVisit(Employee e) {e.setIncome(e.getIncome() + 800);}
}

测试类中修改调用方式

package visitor;import java.util.ArrayList;
import java.util.List;public class test {public static void main(String[] args) {Employee e1 = new Teacher("张三", 8000, 10);Employee e2 = new Engineer("李四", 7000, 20);Visitor visitor = new IncomeVisitor();VacationDaysVisitor vacationDaysVisitor = new VacationDaysVisitor();List<Employee> employeeList = new ArrayList<>();employeeList.add(e1);employeeList.add(e2);for(Employee e : employeeList){e.accept(visitor);e.accept(vacationDaysVisitor);}for(Employee e : employeeList){System.out.println(e);}}
}

在测试类中 

for(Employee e : employeeList){e.accept(visitor);e.accept(vacationDaysVisitor);}

此处调用accept方法进行更新的时候,使用到双分派的技术在选择 e 员工的时候用到一次动态绑定,绑定到真实的员工,在 Engineer和Teacher中选择 visitor 算法的时候再次用到动态绑定,绑定到真实的IncomeVisitor或VacationDaysVisitor,通过两次绑定,即可为合适的员工选择合适的算法。

@Overridepublic void accept(Visitor visitor) {visitor.EngineerVisit(this);}
运行结果如下

如有错误,敬请指正

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

相关文章:

  • Spring框架IOC
  • 有哪些免费的 ERP 软件可供选择?哪些 ERP 软件使用体验较好?
  • 思科CCNA认证都学什么考什么?
  • 模型部署学习笔记——模型部署关键知识点总结
  • 22智能 狄克斯特拉算法复习
  • 首个!艾灵参编的工业边缘计算国家标准正式发布
  • curl也支持断点续传
  • 交换机链路聚合(手动负载分担模式)(eNSP)
  • jmeter 接口性能测试 学习笔记
  • `HashMap`、`Hashtable` 和 `HashSet`的区别
  • Arduino中解析JSON数据
  • linux----文件访问(c语言)
  • 源码分析之Openlayers中MousePosition鼠标位置控件
  • 以ATTCK为例构建网络安全知识图
  • myexcel的使用
  • Unity 上好用的插件
  • Vivado - 远程调试 + 远程综合实现 + vmWare网络配置 + NFS 文件共享 + 使用 VIO 核
  • 双臂机器人
  • 【Lua热更新】上篇
  • Ubuntu批量修改文件名
  • 食家巷大烤馍:岁月沉淀下的麦香传奇
  • harmony UI组件学习(1)
  • BTP Integration Suite CPI Apache Camel
  • vitepress-打包SyntaxError: Element is missing end tag.
  • 【从零开始入门unity游戏开发之——C#篇21】C#面向对象的封装——`this`扩展方法、运算符重载、内部类、`partial` 定义分部类
  • Java进程占用的内存有哪些部分?
  • 【华为OD机试真题】【2024年E卷】数值同化-队列BFS(C++/Java/Python)
  • “魔法糖果盒的秘密:用朴素贝叶斯算法猜糖果颜色”
  • linux中docker命令大全
  • Python `str.strip()` 的高级用法详解