设计模式之访问者模式:一楼千面 各有玄机

news2025/1/4 0:10:25

在这里插入图片描述

~犬📰余~

“我欲贱而贵,愚而智,贫而富,可乎?
曰:其唯学乎”

一、访问者模式概述

\quad 江湖中有一个传说:在遥远的东方,有一座神秘的玉楼。每当武林中人来访,楼中的各个房间都会根据来访者的身份展现出不同的面貌。练剑之人来访,便见剑术精要;习医之人来访,则现医道真谛。一样的楼阁,却能因来访者的不同而呈现万千气象。这,正是访问者模式的真谛。
\quad 在软件设计的世界里,访问者模式就像这座神奇的玉楼。它允许我们将数据结构和数据操作分离,就像将楼阁和访客分开一样。这种设计模式定义了一种方式,让我们能够在不改变已有对象结构的情况下,向其中添加新的操作行为。
\quad 想象一下游乐园的场景:过山车、旋转木马、海盗船等设施早已固定在那里,但每天都会有不同的人来访问它们 —— 游客来游玩、检查员来检修、维护工来保养。每类访问者都会对这些设施进行不同的操作,但设施本身的结构并不会因此改变。

二、访问者模式的角色组成

\quad 在解析访问者模式的角色构成之前,让我们先看一下它的整体结构::

图片
\quad 就像一座精心设计的园林,访问者模式中的每个角色都各司其职,共同构建出一个优雅的结构体系。让我们一起走进这座代码园林,认识一下其中的主要角色:

  • Element(元素):它就像游乐园中的各个游乐设施。每个Element都定义了一个accept方法,这个方法就像设施的接待窗口,来访者必须通过这个窗口才能与设施互动。在我们的游乐园例子中,它可以是过山车、旋转木马等具体设施。
  • Visitor(访问者):它就像是来游乐园的不同人员。可能是来玩耍的游客、来检修的工程师,或是来检查安全的督察员。每种访问者都定义了一系列visit方法,用于访问不同类型的元素。这些方法就像是不同人员对设施的不同操作方式。
  • ObjectStructure(对象结构):像是整个游乐园的管理处。它知道园内有哪些设施,并且负责安排访问者去访问这些设施。当一个安全检查员来到游乐园时,管理处会安排他依次检查所有的设施。
  • ConcreteElement(具体元素):是Element的实现类,就像具体的过山车、旋转木马。它们都实现了accept方法,在方法中通过调用访问者的visit方法来完成具体的操作。这就像每个设施都知道如何配合不同人员的工作。
  • ConcreteVisitor(具体访问者):Visitor的实现类,例如具体的安全检查员、维修工程师等。他们各自实现了visit方法,定义了对不同设施的具体操作流程。检查员检查安全隐患,工程师进行维护保养,各司其职。

\quad 这些角色之间的互动就像是一场精心编排的舞蹈:当游客(ConcreteVisitor)来到游乐园(ObjectStructure)时,管理处会安排他们依次游览各个设施(ConcreteElement)。每个设施都会根据访问者的身份,展现出相应的互动方式。

三、访问者模式案例

\quad 让我们通过一个完整的游乐园管理系统来深入理解访问者模式。在这个系统中,我们需要对不同的游乐设施进行日常检查和维护。每种设施都有其特定的检查点,而不同的工作人员(访问者)也有着不同的工作职责。
\quad 首先,让我们定义设施接口和具体设施:

// 设施接口
public interface Facility {
    void accept(FacilityVisitor visitor);
}

// 过山车设施
public class RollerCoaster implements Facility {
    private String name;
    private int maxSpeed;

    public RollerCoaster(String name, int maxSpeed) {
        this.name = name;
        this.maxSpeed = maxSpeed;
    }

    public String getName() { return name; }
    public int getMaxSpeed() { return maxSpeed; }

    @Override
    public void accept(FacilityVisitor visitor) {
        visitor.visit(this);
    }
}

// 旋转木马设施
public class Carousel implements Facility {
    private String name;
    private int capacity;

    public Carousel(String name, int capacity) {
        this.name = name;
        this.capacity = capacity;
    }

    public String getName() { return name; }
    public int getCapacity() { return capacity; }

    @Override
    public void accept(FacilityVisitor visitor) {
        visitor.visit(this);
    }
}

// 访问者接口
public interface FacilityVisitor {
    void visit(RollerCoaster rollerCoaster);
    void visit(Carousel carousel);
}

// 安全检查员
public class SafetyInspector implements FacilityVisitor {
    @Override
    public void visit(RollerCoaster rollerCoaster) {
        System.out.println("安全检查员正在检查过山车 " + rollerCoaster.getName());
        System.out.println("检查最高速度: " + rollerCoaster.getMaxSpeed() + "km/h");
        System.out.println("检查安全带和刹车系统...");
    }

    @Override
    public void visit(Carousel carousel) {
        System.out.println("安全检查员正在检查旋转木马 " + carousel.getName());
        System.out.println("检查承载人数: " + carousel.getCapacity() + "人");
        System.out.println("检查座椅固定装置...");
    }
}

// 维护工程师
public class MaintenanceEngineer implements FacilityVisitor {
    @Override
    public void visit(RollerCoaster rollerCoaster) {
        System.out.println("维护工程师正在保养过山车 " + rollerCoaster.getName());
        System.out.println("润滑轨道和车轮...");
        System.out.println("检测电机运行状态...");
    }

    @Override
    public void visit(Carousel carousel) {
        System.out.println("维护工程师正在保养旋转木马 " + carousel.getName());
        System.out.println("检查驱动系统...");
        System.out.println("更换磨损零件...");
    }
}

// 游乐园管理类
public class AmusementPark {
    private List<Facility> facilities = new ArrayList<>();

    public void addFacility(Facility facility) {
        facilities.add(facility);
    }

    public void accept(FacilityVisitor visitor) {
        for(Facility facility : facilities) {
            facility.accept(visitor);
        }
    }
}

\quad 现在让我们通过一个具体的例子来运行这个系统:

public class Test{
    public static void main(String[] args) {
        // 创建游乐园
        AmusementPark park = new AmusementPark();

        // 添加设施
        park.addFacility(new RollerCoaster("极速之星", 120));
        park.addFacility(new Carousel("童话木马", 30));

        // 创建访问者
        SafetyInspector inspector = new SafetyInspector();
        MaintenanceEngineer engineer = new MaintenanceEngineer();

        // 进行安全检查
        System.out.println("=== 开始安全检查 ===");
        park.accept(inspector);

        System.out.println("\n=== 开始设备维护 ===");
        park.accept(engineer);
    }
}

\quad 运行这段代码,我们可以看到不同的访问者对相同设施进行不同的操作,而不需要修改设施类的代码:
在这里插入图片描述

四、访问者模式优缺点

4.1. 优点:

\quad 访问者模式最显著的特点是实现了数据结构与数据操作的分离。就像我们的游乐园案例,无论是增加新的检查员还是维护工程师,都不需要修改原有的设施类代码。这种设计非常符合"开闭原则",对扩展开放,对修改关闭。同时,相关的操作行为被集中在访问者类中,使得操作逻辑更加集中和清晰。例如,所有的安全检查逻辑都在SafetyInspector类中,便于统一管理和维护。

4.2. 缺点:

\quad 首先是对扩展元素类型不友好。如果我们要在游乐园中增加一种全新的设施类型,就需要修改所有现有的访问者类,添加相应的visit方法。这违反了"开闭原则"。其次,访问者模式要求元素类的内部结构对访问者是可见的。比如检查员需要知道过山车的最高速度、旋转木马的承载人数等属性,这在某种程度上破坏了对象的封装性。
\quad 此外,使用访问者模式可能会导致系统变得更复杂。我们需要维护多个访问者类,它们之间可能存在一些交叉的职责。比如安全检查和维护工作可能会有重叠的检查项目,这时就需要考虑如何合理划分职责。

五、访问者模式的适用场景

\quad 访问者模式就像是一位经验丰富的管家,最适合处理"对象结构相对稳定,但操作多种多样"的场景。除了我们讨论的游乐园管理系统,它在许多其他领域也有着广泛的应用。
\quad 比如在编译器设计中,语法树的结构一旦确定就不会改变,但我们需要对语法树进行词法分析、语法分析、代码生成等多种操作。又如在文档处理系统中,文档结构(段落、章节、图表等)相对固定,但我们需要对文档进行打印、预览、格式转换等不同操作。
\quad 当你发现系统中有一个复杂的对象结构,而且经常需要对这些对象进行不同的操作时,不妨考虑使用访问者模式。但如果对象结构经常变动,或者操作比较单一,使用访问者模式可能会适得其反。

六、总结

\quad 访问者模式,讲究的是进退有度,相互尊重。它通过巧妙的设计,让数据结构与数据操作得以分离,就像是让每位访客都能以最适合的方式与主人互动。
\quad 在实际应用中,我们要明智地选择是否使用访问者模式。当面对稳定的对象结构和多变的操作需求时,访问者模式如同一位老成持重的管家,能够有条不紊地处理各种访客的需求。但如果对象结构经常变动,或者操作相对单一,使用访问者模式反而会使系统变得臃肿复杂。
\quad 正如古人云:万物有度,过犹不及。设计模式也是如此,关键在于找到最适合当前场景的解决方案。访问者模式,不过是我们设计工具箱中的一件利器,懂得何时使用,方能游刃有余。

在这里插入图片描述

关注犬余,共同进步

技术从此不孤单

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

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

相关文章

结合实例来聊聊UDS诊断中的0x2F服务

1、什么是UDS中的0x2F服务 0x2F简单来说&#xff0c;就是输入输出控制服务。先看官方的简绍 翻译如下&#xff1a; InputOutputControlByldentifier服务来替换输入信号、内部服务器函数和/或强制控制为电子系统的输出&#xff08;执行器&#xff09;的值。通常&#xff0c;此…

1月第二讲:WxPython跨平台开发框架之图标选择界面

1、图标分类介绍 这里图标我们分为两类&#xff0c;一类是wxPython内置的图标资源&#xff0c;以wx.Art_开始。wx.ART_ 是 wxPython 提供的艺术资源&#xff08;Art Resource&#xff09;常量&#xff0c;用于在界面中快速访问通用的图标或位图资源。这些资源可以通过 wx.ArtP…

【弱监督视频异常检测】2024-TCSVT-基于片段间特征相似度的多尺度时间 MLP 弱监督视频异常检测

2024-TCSVT-Inter-clip Feature Similarity based Weakly Supervised Video Anomaly Detection via Multi-scale Temporal MLP 基于片段间特征相似度的多尺度时间 MLP 弱监督视频异常检测摘要1. 引言2. 相关工作A. 分布外检测B. 弱监督视频异常检测C. 多层感知器 3. 方法A. 概述…

C# OpenCV机器视觉:凸包检测

在一个看似平常却又暗藏玄机的午后&#xff0c;阿强正悠闲地坐在实验室里&#xff0c;翘着二郎腿&#xff0c;哼着小曲儿&#xff0c;美滋滋地品尝着手中那杯热气腾腾的咖啡&#xff0c;仿佛整个世界都与他无关。突然&#xff0c;实验室的门 “砰” 的一声被撞开&#xff0c;小…

【英特尔IA-32架构软件开发者开发手册第3卷:系统编程指南】2001年版翻译,2-44

文件下载与邀请翻译者 学习英特尔开发手册&#xff0c;最好手里这个手册文件。原版是PDF文件。点击下方链接了解下载方法。 讲解下载英特尔开发手册的文章 翻译英特尔开发手册&#xff0c;会是一件耗时费力的工作。如果有愿意和我一起来做这件事的&#xff0c;那么&#xff…

8.若依系统监控与定时任务

帮助开发者和运维快速了解应用程序的性能状态。 数据监控 定时任务 实现动态管理任务。 需求&#xff1a;每间隔5s&#xff0c;控制台输出系统时间。 新建的任务类必须在指定目录ruoyi-quartz模块下的task包下。 状态设置为启动 执行策略 场景&#xff1a;比如一个任务每个…

【JAVA高级篇教学】第六篇:Springboot实现WebSocket

在 Spring Boot 中对接 WebSocket 是一个常见的场景&#xff0c;通常用于实现实时通信。以下是一个完整的 WebSocket 集成步骤&#xff0c;包括服务端和客户端的实现。本期做个简单的测试用例。 目录 一、WebSocket 简介 1. 什么是 WebSocket&#xff1f; 2. WebSocket 的特…

【YOLO 项目实战】(12)红外/可见光多模态目标检测

欢迎关注『youcans动手学模型』系列 本专栏内容和资源同步到 GitHub/youcans 【YOLO 项目实战】&#xff08;10&#xff09;YOLO8 环境配置与推理检测 【YOLO 项目实战】&#xff08;11&#xff09;YOLO8 数据集与模型训练 【YOLO 项目实战】&#xff08;12&#xff09;红外/可…

Ubuntu开机The root filesystem on /dev/sdbx requires a manual fsck 问题

出现“Manual fsck”错误可能由以下几种原因引起&#xff1a; 不正常关机&#xff1a;如果系统意外断电或被强制重启&#xff0c;文件系统可能未能正确卸载&#xff0c;导致文件系统损坏。磁盘故障&#xff1a;硬盘的物理损坏可能会引发文件系统错误。文件系统配置问题&#x…

RFSOC 47dr Dp口测试(ARM裸机)

47DR 内核还是一个4核A53的MPSOC&#xff0c;测试方式和MPSOC一样 首先设置好BD文件 编译好BIT设置VITIS工程 examle工程测试即可 但是本人硬件会跑飞不知道为何&#xff0c;通过注释掉下图的子函数得以解决 值得注意的是&#xff0c;最好用HP的线&#xff0c;不要用DP转…

protobuf: 通讯录2.1

先引入需要知道的proto3语法&#xff1a; 1.proto3 1.hexdump 作用&#xff1a; hexdump&#xff1a;是Linux下的⼀个⼆进制⽂件查看⼯具&#xff0c;它可以将⼆进制⽂件转换为ASCII、⼋进制、 ⼗进制、⼗六进制格式进⾏查看。 -C: 表⽰每个字节显⽰为16进制和相应的ASCI…

电子应用设计方案81:智能AI冲奶瓶系统设计

智能 AI 冲奶瓶系统设计 一、引言 智能 AI 冲奶瓶系统旨在为父母或照顾者提供便捷、准确和卫生的冲奶服务&#xff0c;特别是在夜间或忙碌时&#xff0c;减轻负担并确保婴儿获得适宜的营养。 二、系统概述 1. 系统目标 - 精确调配奶粉和水的比例&#xff0c;满足不同年龄段婴…

职场常用Excel基础01-数据验证

大家好&#xff0c;excel在职场中使用非常频繁&#xff0c;今天和大家一起分享一下excel中数据验证相关的内容~ 在Excel中&#xff0c;数据验证&#xff08;Data Validation&#xff09;是一项非常有用的功能&#xff0c;它可以帮助用户限制输入到单元格中的数据类型和范围&am…

Kubernetes Gateway API-3-TLS配置

1 简介 Gateway API 允许使用多种方式配置 TLS。本文档列出了各种TLS设置,并给出了如何有效使用它们的一般指南。 尽管本文档涵盖了 Gateway API 最常见的TLS配置形式,但某些实现也可能提供特定于实现的扩展,允许不同或更高级形式的TLS配置。除此文档外,值得阅读你所使用…

OpenGL入门最后一章观察矩阵(照相机)

前面的一篇文章笔者向大家介绍了模型变化矩阵&#xff0c;投影矩阵。现在只剩下最后一个观察矩阵没有和大家讲了。此片文章就为大家介绍OpenGL入门篇的最后一个内容。 观察矩阵 前面的篇章当中&#xff0c;我们看到了即使没有观察矩阵&#xff0c;我们也能对绘制出来的模型有一…

3.CSS字体属性

3.1字体系列 CSS使用font-family属性定义文本的字体系列。 p{font-family:"微软雅黑"} div{font-family:Arial,"Microsoft Yahei",微软雅黑} 3.2字体大小 css使用font-size属性定义字体大小 p{ font-size:20px; } px(像素)大小是我们网页的最常用的单…

Spring-kafka快速Demo示例

使用Spring-Kafka快速发送/接受Kafka消息示例代码&#xff0c;项目结构是最基础的SpringBoot结构&#xff0c;提前安装好Kafka&#xff0c;确保Kafka已经正确启动 pom.xml&#xff0c;根据个人情况更换springboot、java版本等 <?xml version"1.0" encoding&qu…

R语言统计分析——自助法BOOTSTRAP(1)

参考资料&#xff1a;R语言实战【第2版】 所谓自助法&#xff0c;即从初始样本重复随机替换抽样&#xff0c;生成一个或一系列待检验统计量的经验分布。无需假设一个特定的理论分布&#xff0c;便可生成统计量的置信区间&#xff0c;并能检验统计假设。 举个例子&#xff1a; 我…

yolo数据集格式(txt)转coco格式,方便mmyolo转标签格式

近期使用mmyolo过程中发现工具自带的yolo2coco.py在转换完数据集格式后&#xff0c;可视化标签的时候会有标签错乱情况&#xff0c;具体原因也没找到&#xff0c;肯定是转换过程代码有问题&#xff0c;于是重新做一份代码直接从yolo数据集转化为coco的json格式。 代码如下&…

sonarqube 安装及使用

一、官网参考地址 相关版本下载地址 配置全局变量 .bash_profileexport SONAR_HOME=/Users/jd/soft/sonar-scanner-6.2.1.4610 export PATH=$PATH:$SONAR_HOME/bin export SQ_HOST=http://127.0.0.1:9000/ export SQ_TOKEN=squ_dbb1913e095a92a727a918a9ba6b1af94b007748二、…