对访问者模式的理解
对访问者模式的理解
- 一、场景
- 二、不采用访问者模式
- 1、代码
- 2、特点
- 三、采用访问者模式
- 1、代码
- 2、特点
- 四、思考
一、场景
-
我们有一个图形系统,系统中有多种图形对象(如圆形、方形等),每种图形对象都有不同的属性和行为。现在需要对这些图形对象执行不同的操作,比如计算面积、绘制图形等。
- 图形对象:圆形(Circle)、方形(Square)。
- 操作:计算面积(CalculateArea)、绘制图形(Draw)。
二、不采用访问者模式
1、代码
- 各种图形类
public interface Shape {double calculateArea();void draw();
}public class Circle implements Shape {private double radius;public Circle(double radius) {this.radius = radius;}@Overridepublic double calculateArea() {return Math.PI * radius * radius;}@Overridepublic void draw() {System.out.println("Drawing a circle with radius: " + radius);}
}public class Square implements Shape {private double side;public Square(double side) {this.side = side;}@Overridepublic double calculateArea() {return side * side;}@Overridepublic void draw() {System.out.println("Drawing a square with side: " + side);}
}
- 客户端
public class Main {public static void main(String[] args) {Shape circle = new Circle(5);Shape square = new Square(4);System.out.println("Circle area: " + circle.calculateArea());circle.draw();System.out.println("Square area: " + square.calculateArea());square.draw();}
}/*
Circle area: 78.53981633974483
Drawing a circle with radius: 5.0
Square area: 16.0
Drawing a square with side: 4.0
*/
2、特点
我这里没说缺点,因为在当前的情况下,上述设计是不错的。
-
上述设计的思路是各种图形实现各自的行为,例如:圆形和方形各自实现计算面积的算法。
-
但随着业务的发展,原本小而美的类,会出现越来越多的方法。慢慢的,类不再是单一职责了。
- 例如:我们还需要计算图形的周长。
三、采用访问者模式
1、代码
-
各种图形类
public interface Shape {void accept(ShapeVisitor visitor); }public class Circle implements Shape {private double radius;public Circle(double radius) {this.radius = radius;}@Overridepublic void accept(ShapeVisitor visitor) {visitor.visitCircle(this);}public double getRadius() {return radius;} }public class Square implements Shape {private double side;public Square(double side) {this.side = side;}@Overridepublic void accept(ShapeVisitor visitor) {visitor.visitSquare(this);}public double getSide() {return side;} }- 图形的行为外派给ShapeVistor
-
各种访问者
public interface ShapeVisitor {void visitCircle(Circle circle);void visitSquare(Square square); }public class DrawVisitor implements ShapeVisitor {@Overridepublic void visitCircle(Circle circle) {System.out.println("Drawing a circle with radius: " + circle.getRadius());}@Overridepublic void visitSquare(Square square) {System.out.println("Drawing a square with side: " + square.getSide());} }public class AreaVisitor implements ShapeVisitor {@Overridepublic void visitCircle(Circle circle) {double area = Math.PI * circle.getRadius() * circle.getRadius();System.out.println("Circle area: " + area);}@Overridepublic void visitSquare(Square square) {double area = square.getSide() * square.getSide();System.out.println("Square area: " + area);} } -
客户端
public class Main {public static void main(String[] args) {Shape circle = new Circle(5);Shape square = new Square(4);ShapeVisitor areaVisitor = new AreaVisitor();ShapeVisitor drawVisitor = new DrawVisitor();circle.accept(areaVisitor);circle.accept(drawVisitor);square.accept(areaVisitor);square.accept(drawVisitor);} }/* Circle area: 78.53981633974483 Drawing a circle with radius: 5.0 Square area: 16.0 Drawing a square with side: 4.0 */
2、特点
-
假设要计算图形的周长,我们新增一个PerimeterVisitor即可。
- 如果不采用访问者模式,我们需要给Circle、Square这两个类各自新增计算周长的方法。
- 显然,访问者模式更遵循开闭原则。
四、思考
-
在实际开发中,我们先定义接口,再定义实现类。往往会面临一个尴尬地处境:接口中的方法越加越大,实现类也越发臃肿。
-
这个时候,访问者模式会发挥一定的作用:想明白接口中哪些方法是这个接口必须的,哪些方法是随着业务发展不断扩展的。将扩展的方法用访问者模式实现。
// 在访问器模式中,这种也叫Element接口。 public interface InterfaceA {// 必须的方法void methodA();// 扩展方法void accept(Vistor vistor); }- 很显然,传入不同的vistor,就实现了不同的扩展。
-
这时候有人可能会说:增加一种类型,Vistor接口也会增加方法啊。Vistor接口的方法也可能越来越多啊。
-
这时候就要具体问题具体分析了,
- 情况1:如果具体的Element随着业务的发展越来越多,但Element接口的方法不怎么增加,显然,不采用访问者模式更好。
- 情况2:但如果具体的Element在软件设计时确定下来了,后续也不怎么增加了,但每个Element中的方法会越来越多,显然,采用访问者模式更好。
-
-
假设我们遇到的是情况2,采用了访问者模式进行软件设计,正在写如下代码:
public class AreaVisitor implements ShapeVisitor {@Overridepublic void visitCircle(Circle circle) {// 现有的Circle的方法不足以实现这个功能,没办法,得给Circle增加方法。} }- 如果我们在实现Vistor时,强依赖Element的方法,那么说明这个方法不适合由Vistor来实现,因为Circle提供一个public方法,结果只给visitCircle方法用,这是不合理的(不够封装)。此时,可以将该方法挪到Element接口中。
-
假设其他人设计了访问者模式,我们来接手开发,当我们要实现一个新需求的时候,我们既可以在具体Element类中实现方法来满足诉求,又可以实现一个XxxVisitor类来满足诉求。
-
例如:要计算图形的周长
- 选择1:我们可以在Shape接口中新增计算周长的方法,然后Circle和Square去实现这个方法。
- 选择2:我们也可以新增一个PerimeterVisitor类,在这个类中新增2个方法,一个给Circle计算周长,另一个给Square计算周长。
-
每个人对设计模式的了解程度是不同的, 必然会出现有的人按选择1进行开发,有的人按选择2进行开发。慢慢地,这些代码变成了屎山。
-
-
经过上述思考,我个人认为访问者模式并不是好的设计模式。
- 访问者模式是一种行为设计模式, 它能将算法与其所作用的对象隔离开来。
- 但事实是,Vistor接口的实现还是依赖于具体的Element。算法还是没法和对象真正隔开。
相关文章:
对访问者模式的理解
对访问者模式的理解 一、场景二、不采用访问者模式1、代码2、特点 三、采用访问者模式1、代码2、特点 四、思考 一、场景 我们有一个图形系统,系统中有多种图形对象(如圆形、方形等),每种图形对象都有不同的属性和行为。现在需要对…...
医疗机构中核心业务相关的IT设备全面解析
一、引言 在医疗行业数字化转型的进程中,IT设备作为医疗机构核心业务运行的基础设施,其重要性愈发凸显。医疗机构的核心业务涵盖患者诊疗、临床管理、检验检测、影像诊断、药品管理、电子病历、医院信息系统(HIS)、实验室信息系统…...
【Vue】b站黑马视频学习笔记(导览)
Vue学习导览 1.Vue基础知识>> 2.组件>> 3.路由>> 4.路由案例-面经基础版>> 5. vuex>> 6.购物车案例>>...
AI前端组件库Ant DesIgn X
Ant Design X AI:体验新秩序 Ant Design 团队精心打造 RICH 设计范式,为 AI 界面提供卓越解决方案,引领智能交互新体验。 设计语言与理论 官网: Ant Design X - 轻松打造 AI 驱动的界面。 AI 设计范式 —— RICH 是我们在蚂蚁…...
【C++编程基础-关键字】:define和inline的区别
一、引言:开启探索之旅 在 C++ 编程的广阔天地中,define和inline就如同两把独特的钥匙,各自解锁着不同的代码优化与功能实现的大门。define作为 C/C++ 语言中的预处理指令,有着悠久的历史,它能够实现文本替换,在代码编译前就发挥着重要作用,为我们带来了诸如定义常量、…...
追踪大型语言模型的思维过程:提示词工程重要
追踪大型语言模型的思维过程:提示词工程重要 目录 追踪大型语言模型的思维过程:提示词工程重要**1. 分步思考能力:像人类一样打草稿****2. 跨语言概念词典:突破语言符号的束缚****3. 诗歌押韵规划:神经元提前预留韵脚****4. 编造专业解释:数据模式导致的“客服式回应”**…...
BGP路由协议之属性1
公认属性是所有 BGP 路由器都必须能够识别的属性 公认必遵 (Well-known Mandatory) : 必须包括在每个 Update 消息里公认任意 (Well-known Discretionary) : 可能包括在某些 Update 消息里。 可选属性不需要都被 BGP 路由器所识别 可选过渡(OptionalTransitive) : BGP 设备不…...
什么是 k8s 的 Taints(污点) 和 Tolerations(容忍度)
什么是 k8s 的 Taints(污点) 和 Tolerations(容忍度) 在 Kubernetes(K8s)中,Taints(污点)和 Tolerations(容忍度)用于影响 Pod 调度到节点的行为…...
C++类模板的运用
使用vector实现一个简单的本地注册登录系统 注册:将账号密码存入vector里面,注意防重复判断 登录:判断登录的账号密码是否正确 #include <iostream> #include <vector> #include <string> #include <algorithm>us…...
人工智能:深度学习关键技术与原理详解
深度学习作为机器学习的核心分支,通过构建多层神经网络实现对复杂数据的高效建模。其关键技术及原理可分为以下几个方面: 一、关键技术 神经网络结构 多层感知机(MLP):基础的前馈神经网络,通过全连接层堆叠…...
coze生成流程图和思维导图工作流
需求:通过coze平台实现生成流程图和思维导图,要求支持文档上传 最终工作流如下: 入参: 整合用户需求文件内容的工作流:https://blog.csdn.net/YXWik/article/details/147040071 选择器分发,不同的类型走…...
【数据库】达梦arm64安装
话不多说,快速开始~ 1.下载 进入官网: 产品下载 | 达梦在线服务平台 下载安装包。 选飞腾、鲲鹏都可以,都是arm架构的。我选择的是: 直接下载地址是https://download.dameng.com/eco/adapter/DM8/202502/dm8_20250117_HWarm920…...
AR 赋能儿童娱乐:剧本杀与寻宝小程序搭建秘籍
在科技飞速发展的当下,儿童娱乐领域正经历着一场创新变革。AR(增强现实)技术的融入,为儿童剧本杀与寻宝游戏带来了前所未有的沉浸式体验。通过搭建专属小程序,孩子们能够在虚拟与现实交织的世界中开启奇幻冒险。接下来…...
交换机可以代替路由器的功能吗
交换机在一定程度上可以承担部分路由器的功能,但不能完全代替路由器,原因如下: 1、工作层次不一样: a、交换机工作在数据链路层,主要功能是基于MAC地址(物理地址)进行数据帧的转发,…...
leetcode274.H指数
直接排序完后进行遍历 class Solution {public int hIndex(int[] citations) {Arrays.sort(citations);int result 0;for (int i citations.length-1; i >0; i--) {if(citations[i]>citations.length-i)resultcitations.length-i;elsebreak;}return result;} }...
人工智能基础知识详解:从概念到前沿技术与应用
在数字化浪潮席卷全球的今天,人工智能(Artificial Intelligence,简称AI)已不再是科技前沿的神秘概念,而是融入我们日常工作的实用工具。从智能语音助手到自动驾驶汽车,从医疗影像诊断到生成式艺术创作&…...
JavaScript | 对象
对象 JS中,对象是一组无序的相关属性和方法的集合,所有的事物都是对象,例如:字符串、数值、数组、函数等 对象是由属性与方法组成的 属性:事物的特征,在对象中用属性来表示(一般用名词)方法:…...
内网文件传输新体验,聊天、传输、自定义,一应俱全
Flix 是一款高效、便捷的跨平台局域网文件传输工具,支持 Windows、macOS、Android、iOS 和 Linux 等多种操作系统。它以简洁直观的聊天式界面为特色,让用户能够像发送消息一样轻松地传输文件,无需复杂的设置或登录。Flix 支持大文件和多种格式…...
Vue PDF Annotation plugin library online API examples
This article introduces the online version of the ElasticPDF API tutorial for the PDF annotation plug-in library in Vue projects. The API includes ① Export edited PDF data; ② Export annotations json data; ③ Reload old annotations; ④ Change files; ⑤ Se…...
Go 学习笔记 · 进阶篇 · 第一天:接口与多态
🐶Go接口与多态:继承没了,但自由炸裂! 最近翻 Go 的代码,突然看到这么一段: type Animal interface {Speak() string }我一愣,咦?这不就是 Java 里常见的“接口”吗? …...
C语言传参寄存器压栈流程总结
相关 《Linux函数调用栈的实现原理(X86)》 总结 rsp向低地址生长(栈顶),rbp记录旧值(栈底)。 intel x86测试,六个和六个以内的参数用寄存器传递。8个参数场景,6个用寄存…...
C盘清理——快速处理
C盘清理 | 快速处理 软件:小番茄C盘清理 https://ccleancdn.xkbrowser.com/cleanmaster/FanQieClean_13054_st.exe 前言:为什么需要专业的C盘清理工具? 作为一位长期与Windows系统打交道的技术博主,我深知C盘空间不足带来的痛苦…...
前端服务配置详解:从入门到实战
前端服务配置详解:从入门到实战 一、环境配置文件(.env) 1.1 基础结构 在项目根目录创建 .env 文件: # 开发环境 VUE_APP_API_BASE_URL http://localhost:3000/api VUE_APP_VERSION 1.0.0# 生产环境(.env.produc…...
历年跨链合约恶意交易详解(四)——Chainswap20210711
漏洞合约函数 function receive(uint256 fromChainId, address to, uint256 nonce, uint256 volume, Signature[] memory signatures) virtual external payable {_chargeFee();require(received[fromChainId][to][nonce] 0, withdrawn already);uint N signatures.length;r…...
Python基于OpenCV和SVM实现中文车牌识别系统GUI界面
说明:这是一个系统实战项目,如需项目代码可以直接到文章最后关注获取。 项目背景 随着智能交通系统和智慧城市的发展,车牌识别技术在车辆管理、交通监控、停车场收费等领域发挥着重要作用。传统的车牌识别系统主要针对英文和数字的识别&…...
有瓶颈设备的多级生产计划问题:基于Matlab的深度解析与实践
内容摘要 本文围绕有瓶颈设备的多级生产计划问题展开,通过实例详细阐述问题背景、建立数学模型,并用Matlab代码进行求解。旨在帮助读者理解该问题的本质,掌握利用Matlab解决此类生产计划优化问题的方法,为企业在实际生产中合理规…...
网络性能优化参数关系解读 | TCP Nagle / TCP_NODELAY / TCP_QUICKACK / TCP_CORK
注:本文为 “网路性能优化” 相关文章合辑。 未整理去重。 如有内容异常,请看原文。 TCP_NODELAY 详解 lenky0401 发表于 2012-08-25 16:40 在网络拥塞控制领域,Nagle 算法(Nagle algorithm)是一个非常著名的算法&…...
mac命令操作
mac命令操作 快速删除一行: control u 剪切文件:步骤1、先进行Command c 进行选择复制文件,2、进行commandoptionv进行移动文件,如果commandv是进行复制文件。 commandcontrolD 三个键即可屏幕取词进行翻译 mac中可以使用快捷方…...
react 18 可中断的理解以及应用
React 的“可中断(interruptible)”渲染,指的是 React 在执行渲染过程中可以暂停、中断、再继续或放弃更新。这是 React 18 引入的并发特性的一部分,目的是让界面响应更流畅,防止“卡顿”。 📖 举个例子&am…...
【Python】Python环境管理工具UV安装gdal
目录 一、UV简介1.2 UV高效包管理工具二、UV配置流程步骤1:安装UV工具步骤2:配置环境变量(Windows)三、UV包管理实战3.1 常用命令速查3.2 完整 `uv` 工作流(无需手动 `venv`)**1. 创建项目****2. 初始化依赖管理(可选)****3. 添加依赖****4. 运行代码****5. 更新/移除依…...
