访问者模式(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 系统展示 系统首页界面图 用户注册界面图 服务信息界面…

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…

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…

搭建谷歌Gemini

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

一张图让你学会Python

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

C++类和对象一

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

③【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库是无法运行…

第一次vp蓝桥杯

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

【Flutter 面试题】Flutter如何处理响应式布局?

【Flutter 面试题】Flutter如何处理响应式布局&#xff1f; 文章目录 写在前面口述回答补充说明完整代码示例运行结果详细说明 写在前面 &#x1f64b; 关于我 &#xff0c;小雨青年 &#x1f449; CSDN博客专家&#xff0c;GitChat专栏作者&#xff0c;阿里云社区专家博主&am…

NB-IoT模块

目录 一. NB-IoT模块实物图 二. BC20/NB-IoT模块产品规格 三. 指令顺序 1. AT判断BC20模组是否正常 2. ATE0返回OK&#xff0c;已经返回回显 3. ATCSQ 4. AT_CEREG? 5. ATCGATT? 6. ATCGATT? 四. OneNet 连接 1. AT 查看 NB(当前NB)&#xff0c;云平台根据这两个…

从顺序表到链表再到队列和栈

1.顺序表 顺序表&#xff0c;简单的说&#xff0c;就是一种用结构体储存的数组。只是一般顺序表还有着记录存入数据个数size和数组总空间位置个数capacity 我们要定义一个顺序表的结构体&#xff0c;就要先确定顺序表的储存的数据&#xff0c;然后假设数组是固定长度&am…

链表的分类有哪些?

1、典型回答 链表 (Linked List) 是一种常见的线性数据结构&#xff0c;由一系列节点(Node)组成。每个节点都包含数据(element) 和一个指向下一个节点的指针 (next) 。通过这种方式&#xff0c;每个节点可以按照顺序链接在一起&#xff0c;形成一个链表。 线性数据结构是一种常…

ArcGIS学习(十五)用地适宜性评价

ArcGIS学习(十五)用地适宜性评价 本任务给大家带来的内容是用地适宜性评价。 用地适宜性评价是大家在平时工作中最常接触到的分析场景之一。尤其是在国土空间规划的大背景下,用地适宜性评价变得越来越重要。 此外,我们之前的任务主要是使用矢量数据进行分析。本案例是主讲…

Java中的 “==” 与 equals 的区别

Java中的 “” 与 equals 的区别 1.“” 在Java中有两大类数据类型&#xff0c;一类是基础数据类型共有八种分别是byte、 short、 int、 long、 float、 double、 char、boolean&#xff0c;另一类则是引用数据类型&#xff0c;例如String、Integer等等。 “ ” 作为比较运算…

切片上的健壮范型函数

在这篇博客文章中&#xff0c;我们将讨论如何通过了解切片在内存中的表示方式以及这对垃圾收集器的影响&#xff0c;更有效地使用slices包中提供的函数。我们还将介绍我们最近如何调整这些函数&#xff0c;使它们变得不那么令人惊讶。 借助类型参数&#xff0c;我们可以为所有…