访问者模式(Visitor Pattern)

访问者模式

说明

访问者模式(Visitor Pattern)属于行为型模式,表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。
该模式是将数据结构与数据操作分离的设计模式,是将作用于某种数据结构中的各元素的操作封装起来。该模式的实施前提是数据结构相对稳定,不会频繁变动,但是作用于结构上的操作又经常变动。

结构

访问者模式的主要角色如下:
抽象访问者(Visitor):即作用于元素操作的抽象,定义了访问具体元素的接口;
具体访问者(Concrete Visitor):根据需求实现对具体元素的操作;
抽象元素(Element):被访问的元素抽象,定义了接收访问者的接口;
具体元素(Concrete Element):被访问的具体元素,实现接收方法;
结构对象(Object Struture):维护了具体元素的集合,并提供方法接收访问者对集合中的元素进行访问。
访问者模式-类结构图

代码案例

业务背景是:某个培训机构只有语文、英语、数学老师,每个科目的老师都有对应的行为,并且行为会随着需求而改变。

反例

/*** @program: visitor* @description: 老师抽象类* @author: wxw* @create: 2024-03-14 16:12**/
public abstract class Teacher {// 姓名protected String name;public Teacher(String name) {this.name = name;}// 备课public abstract void prepareLessons();// 上课public abstract void giveClass();// 布置作业public abstract void assignHomework();// 批改作业public abstract void homeworkCorrecting();
}
/*** @program: visitor* @description: 语文老师* @author: wxw* @create: 2024-03-14 16:15**/
public class ChineseTeacher extends Teacher{public ChineseTeacher(String name) {super(name);}@Overridepublic void prepareLessons() {System.out.println(String.format("语文老师【%s】开始备课.", this.name));}@Overridepublic void giveClass() {System.out.println(String.format("语文老师【%s】开始授课.", this.name));}@Overridepublic void assignHomework() {System.out.println(String.format("语文老师【%s】开始布置作业.", this.name));}@Overridepublic void homeworkCorrecting() {System.out.println(String.format("语文老师【%s】开始批改作业.", this.name));}}
/*** @program: visitor* @description: 英语老师* @author: wxw* @create: 2024-03-14 16:15**/
public class EnglishTeacher extends Teacher{public EnglishTeacher(String name) {super(name);}@Overridepublic void prepareLessons() {System.out.println(String.format("英语老师【%s】开始备课.", this.name));}@Overridepublic void giveClass() {System.out.println(String.format("英语老师【%s】开始授课.", this.name));}@Overridepublic void assignHomework() {System.out.println(String.format("英语老师【%s】开始布置作业.", this.name));}@Overridepublic void homeworkCorrecting() {System.out.println(String.format("英语老师【%s】开始批改作业.", this.name));}}
/*** @program: visitor* @description: 数学老师* @author: wxw* @create: 2024-03-14 16:15**/
public class MathTeacher extends Teacher{public MathTeacher(String name) {super(name);}@Overridepublic void prepareLessons() {System.out.println(String.format("数学老师【%s】开始备课.", this.name));}@Overridepublic void giveClass() {System.out.println(String.format("数学老师【%s】开始授课.", this.name));}@Overridepublic void assignHomework() {System.out.println(String.format("数学老师【%s】开始布置作业.", this.name));}@Overridepublic void homeworkCorrecting() {System.out.println(String.format("数学老师【%s】开始批改作业.", this.name));}}
public class Test {public static void main(String[] args) {List<Teacher> teachers = new ArrayList<>();Teacher zhangsan = new ChineseTeacher("张三");Teacher lisi = new MathTeacher("李四");Teacher wangwu = new EnglishTeacher("王五");teachers.add(zhangsan);teachers.add(lisi);teachers.add(wangwu);for (Teacher t: teachers) {System.out.println("============================");t.prepareLessons();t.giveClass();t.assignHomework();t.homeworkCorrecting();System.out.println("============================");}}
}

假设培训机构为了提高教学质量,要求所有老师都必须跟同学进行单独沟通,了解每个同学的情况。
这时就需要在Teacher抽象类中添加沟通的方法,所有的子类都得实现该方法,违背了开闭原则。

重构后

抽象元素(Element)

/*** @program: visitor* @description: 老师抽象类*              抽象元素(Element)* @author: wxw* @create: 2024-03-14 16:12**/
public abstract class Teacher {// 姓名protected String name;public Teacher(String name) {this.name = name;}// 接收方法public abstract void accept(IActionVisitor actionVisitor);
}

抽象访问者(Visitor)

/*** @program: visitor* @description: 老师行为访问者接口*              抽象访问者(Visitor)* @author: wxw* @create: 2024-03-14 16:33**/
public interface IActionVisitor {// 访问语文老师void visitorChineseTeacher(ChineseTeacher chineseTeacher);// 访问英语老师void visitorEnglishTeacher(EnglishTeacher englishTeacher);// 访问数学老师void visitorMathTeacher(MathTeacher mathTeacher);}

具体元素(Concrete Element)

/*** @program: visitor* @description: 语文老师*              具体元素(Concrete Element)* @author: wxw* @create: 2024-03-14 16:15**/
public class ChineseTeacher extends Teacher {public ChineseTeacher(String name) {super(name);}@Overridepublic void accept(IActionVisitor actionVisitor) {actionVisitor.visitorChineseTeacher(this);}}/*** @program: visitor* @description: 英语老师*              具体元素(Concrete Element)* @author: wxw* @create: 2024-03-14 16:15**/
public class EnglishTeacher extends Teacher {public EnglishTeacher(String name) {super(name);}@Overridepublic void accept(IActionVisitor actionVisitor) {actionVisitor.visitorEnglishTeacher(this);}}/*** @program: visitor* @description: 数学老师*              具体元素(Concrete Element)* @author: wxw* @create: 2024-03-14 16:15**/
public class MathTeacher extends Teacher {public MathTeacher(String name) {super(name);}@Overridepublic void accept(IActionVisitor actionVisitor) {actionVisitor.visitorMathTeacher(this);}
}

具体访问者(Concrete Visitor)

/*** @program: visitor* @description: 备课访问者*              具体访问者(Concrete Visitor)* @author: wxw* @create: 2024-03-14 16:37**/
public class PrepareLessonsVisitor implements IActionVisitor {@Overridepublic void visitorChineseTeacher(ChineseTeacher chineseTeacher) {System.out.println(String.format("语文老师【%s】开始备课.", chineseTeacher.name));}@Overridepublic void visitorEnglishTeacher(EnglishTeacher englishTeacher) {System.out.println(String.format("英语老师【%s】开始备课.", englishTeacher.name));}@Overridepublic void visitorMathTeacher(MathTeacher mathTeacher) {System.out.println(String.format("数学老师【%s】开始备课.", mathTeacher.name));}
}/*** @program: visitor* @description: 上课访问者*              具体访问者(Concrete Visitor)* @author: wxw* @create: 2024-03-14 16:37**/
public class GiveClassVisitor implements IActionVisitor {@Overridepublic void visitorChineseTeacher(ChineseTeacher chineseTeacher) {System.out.println(String.format("语文老师【%s】开始上课.", chineseTeacher.name));}@Overridepublic void visitorEnglishTeacher(EnglishTeacher englishTeacher) {System.out.println(String.format("英语老师【%s】开始上课.", englishTeacher.name));}@Overridepublic void visitorMathTeacher(MathTeacher mathTeacher) {System.out.println(String.format("数学老师【%s】开始上课.", mathTeacher.name));}
}/*** @program: visitor* @description: 布置作业访问者*              具体访问者(Concrete Visitor)* @author: wxw* @create: 2024-03-14 16:37**/
public class AssignHomeworkVisitor implements IActionVisitor {@Overridepublic void visitorChineseTeacher(ChineseTeacher chineseTeacher) {System.out.println(String.format("语文老师【%s】开始布置作业.", chineseTeacher.name));}@Overridepublic void visitorEnglishTeacher(EnglishTeacher englishTeacher) {System.out.println(String.format("英语老师【%s】开始布置作业.", englishTeacher.name));}@Overridepublic void visitorMathTeacher(MathTeacher mathTeacher) {System.out.println(String.format("数学老师【%s】开始布置作业.", mathTeacher.name));}
}/*** @program: visitor* @description: 批改作业访问者*              具体访问者(Concrete Visitor)* @author: wxw* @create: 2024-03-14 16:37**/
public class HomeworkCorrectingVisitor implements IActionVisitor {@Overridepublic void visitorChineseTeacher(ChineseTeacher chineseTeacher) {System.out.println(String.format("语文老师【%s】开始批改作业.", chineseTeacher.name));}@Overridepublic void visitorEnglishTeacher(EnglishTeacher englishTeacher) {System.out.println(String.format("英语老师【%s】开始批改作业.", englishTeacher.name));}@Overridepublic void visitorMathTeacher(MathTeacher mathTeacher) {System.out.println(String.format("数学老师【%s】开始批改作业.", mathTeacher.name));}
}

结构对象(Object Struture)

/*** @program: visitor* @description: 培训机构对象结构*              结构对象(Object Struture)* @author: wxw* @create: 2024-03-14 16:43**/
public class TrainingAgency {private List<Teacher> teacherList = new ArrayList<>();public void addTeacher(Teacher teacher){teacherList.add(teacher);}public void removeTeacher(Teacher teacher){teacherList.remove(teacher);}public void startWork(IActionVisitor visitor){System.out.println("================================");for (Teacher t: teacherList) {t.accept(visitor);}System.out.println("================================");}
}

客户端

public class Test {public static void main(String[] args) {TrainingAgency trainingAgency = new TrainingAgency();trainingAgency.addTeacher(new ChineseTeacher("张三"));trainingAgency.addTeacher(new EnglishTeacher("李四"));trainingAgency.addTeacher(new MathTeacher("王五"));// 备课访问者IActionVisitor prepareLessonsVisitor = new PrepareLessonsVisitor();trainingAgency.startWork(prepareLessonsVisitor);// 上课访问者IActionVisitor giveClassVisitor = new GiveClassVisitor();trainingAgency.startWork(giveClassVisitor);// 布置作业访问者IActionVisitor assignHomeworkVisitor = new AssignHomeworkVisitor();trainingAgency.startWork(assignHomeworkVisitor);// 批改作业访问者IActionVisitor homeworkCorrectingVisitor = new HomeworkCorrectingVisitor();trainingAgency.startWork(homeworkCorrectingVisitor);}
}

增加沟通访问者,无需修改Teacher相关的类

/*** @program: visitor* @description: 与学生沟通访问者*              具体访问者(Concrete Visitor)* @author: wxw* @create: 2024-03-14 16:37**/
public class CommunicationVisitor implements IActionVisitor {@Overridepublic void visitorChineseTeacher(ChineseTeacher chineseTeacher) {System.out.println(String.format("语文老师【%s】开始与学生沟通.", chineseTeacher.name));}@Overridepublic void visitorEnglishTeacher(EnglishTeacher englishTeacher) {System.out.println(String.format("英语老师【%s】开始与学生沟通.", englishTeacher.name));}@Overridepublic void visitorMathTeacher(MathTeacher mathTeacher) {System.out.println(String.format("数学老师【%s】开始与学生沟通.", mathTeacher.name));}
}

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

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

相关文章

Flutter 多语言、主题切换之GetX库

多语言、主题切换之GetX库 前言正文一、配置项目二、模拟UI三、语言配置① 常量键② 语言配置文件③ 配置 四、持久化五、切换语言① my_home.dart② home.dart③ mine_controller.dart④ language_setting_controller.dart⑤ language_setting.dart⑥ mine.dart 六、切换主题①…

基于SpringBoot的“家政服务管理平台”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“家政服务管理平台”的设计与实现&#xff08;源码数据库文档PPT) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBoot 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 系统首页界面图 用户注册界面图 服务信息界面…

DM_SQL

DM_SQL 语言符合结构化查询语言 SQL 标准&#xff0c;是标准 SQL 的扩充。它集数据定义、数据查询、数据操纵和数据控制于一体&#xff0c;是一种统一的、综合的关系数据库语言。它功能强大&#xff0c;使用简单方便、容易为用户掌握。DM_SQL 语言具有如下特点&#xff1a; 1.…

WorldGPT、Pix2Pix-OnTheFly、StyleDyRF、ManiGaussian、Face SR

本文首发于公众号&#xff1a;机器感知 WorldGPT、Pix2Pix-OnTheFly、StyleDyRF、ManiGaussian、Face SR HandGCAT: Occlusion-Robust 3D Hand Mesh Reconstruction from Monocular Images We propose a robust and accurate method for reconstructing 3D hand mesh from m…

HDOJ 2045

不容易系列之(3)—— LELE的RPG难题 Problem Description 人称“AC女之杀手”的超级偶像LELE最近忽然玩起了深沉&#xff0c;这可急坏了众多“Cole”&#xff08;LELE的粉丝,即"可乐"&#xff09;,经过多方打探&#xff0c;某资深Cole终于知道了原因&#xff0c;原来…

ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接。

发生的错误信息&#xff1a; File "C:\Users\malongqiang\.conda\envs\ObjectDetection\lib\ssl.py", line 1309, in do_handshakeself._sslobj.do_handshake() ConnectionResetError: [WinError 10054] 远程主机强迫关闭了一个现有的连接。 分析原因&#xff1a; …

接口的回调

接口是引用型变量 接口回调技术&#xff1a;把实现该接口的类的实例引用赋给接口变 量&#xff08;存放对象的引用&#xff09;→接口变量可调用被类重写的接口方法。 ★注意★&#xff1a;接口com无法调用类中非接口方法

SQL语句在MySQL中的执行过程

有一条SQL语句&#xff0c;给到MySQL,是怎么被执行的 基本架构 连接器&#xff1a;进行身份认证&#xff0c;确定操作权限 查询缓存&#xff1a; 执行查询语句时&#xff0c;先查询缓存&#xff08;不太实用&#xff0c;MySQL 8.0 版本后删了&#xff09; 分析器&#xff1a…

格子表单GRID-FORM | 必填项检验 BUG 修复实录

格子表单/GRID-FORM已在Github 开源&#xff0c;如能帮到您麻烦给个星&#x1f91d; GRID-FORM 系列文章 基于 VUE3 可视化低代码表单设计器嵌套表单与自定义脚本交互文档网站搭建&#xff08;VitePress&#xff09;与部署&#xff08;Github Pages&#xff09;必填项检验 BUG…

牛客小白月赛61-E-排队

很好的一道题啊,学到了不少东西!!!! 首先是一个结论 逆序对总数 n! / 2 * 不相等的数字对数 (1) 不相等的数字对数怎么求 结论 不相等的数字对数 C(n,2) - ∑C(2,cnt(i))(i数字的出现次数) (2) n! / 2 怎么处理,有取模的除运算怎么处理??? 这块一直不会,今…

20240314 大模型快讯

//社区生态// 零一万物发布Yi大模型API开放平台。零一万物通过开放 API 让包括开发者在内的更多人用上强大的对话、多模态大模型&#xff0c;用来创作或用于工作。 //智能体// 全球首个OpenAI机器人诞生。初创公司Figure联合OpenAI&#xff0c;推出名为Figure 01的机器人&…

搭建谷歌Gemini

前言 Gemini是Google AI于2023年发布的大型语言模型&#xff0c;拥有强大的文本生成、理解和转换能力。它基于Transformer模型架构&#xff0c;并使用了大量文本和代码数据进行训练。Gemini可以执行多种任务&#xff0c;包括&#xff1a; 生成文本&#xff1a;可以生成各种类…

一张图让你学会Python

有编程基础的人一看就可以了解 Python 的用法了。真正的 30 分钟上手。 国外一高手画的&#xff0c;现把它翻译成中文&#xff0c;入门超简单 python入门神图!(看不清可以“另存为”查看)

go的slice学习

并发访问slice 线上出现一粒多协程并发append全局slice的情况&#xff0c;导致内存不断翻倍&#xff0c;因此对slice的使用需要重新考虑。 并发读写的情况下&#xff0c; 可以利用锁、channel等避免竞态 问题 func TestDemo32(t *testing.T) {var wg sync.WaitGroupvar n 1…

C++类和对象一

#include <iostream> using namespace std;//设计一个学生类 class CStudent {public: //公有成员void InputData(){cout << "请输入学号";cin >> sno;cout << "请输入姓名";cin >> sname;cout << "请输入分…

NAT笔记

NAT 用于实现私网和公网之间的互访。 静态NAT 静态NAT实现私网地址和公网地址的一对一转换。 有2种配置方法&#xff1a; 全局模式下设置静态NAT [R1]nat static global 172.10.10.10 inside 192.168.10.10 [R1]int g0/0/1 #外网出口 [R1-GigabitEthernet0/0/1]nat static…

③【Docker】Docker部署Nginx

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ ③【Docker】Docker部署Nginx docker拉取nginx…

Linux环境下安装DGL及其CUDA

前段时间看到一篇AAAI2024的论文Patch-wise Graph Contrastive Learning for Image Translation&#xff0c;它采用GNN的思想来进行image-to-image translation的任务&#xff0c;非常的新颖&#xff0c;但我进行复现的时候&#xff0c;发现直接下载它里面需要的DGL库是无法运行…

MyBatis 之三:配置文件详解和 Mapper 接口方式

配置文件 配置文件中常用的元素&#xff1a; configuration 配置 properties 可以配置在Java 属性配置文件中 settings 修改 MyBatis 在运行时的行为方式 typeAliases 为 Java 类型命名一个别名&#xff08;简称&#xff09; typeHandlers 类型处理器 objectFactory 对象工厂 p…

第一次vp蓝桥杯

最失败的一集&#xff0c;这是学了个什么&#xff1f; 果然是一个很失败的人呢&#xff0c;第一次逃晚自习就被辅导员发现了呢&#xff0c;还给我打电话&#xff0c;虽然知道可能他也没办法。但这就更体现我很失败了。 题也不会写&#xff0c;其他的方面也不是很如意。嘻嘻嘻…