趣学设计模式之访问者模式
创作时间:
作者:
@小白创作中心
趣学设计模式之访问者模式
引用
CSDN
1.
https://blog.csdn.net/qq_56158663/article/details/146021887
目录
- 一、 啥是访问者模式?
- 二、 为什么要用访问者模式?
- 三、 访问者模式的实现方式
- 四、 访问者模式的优缺点
- 五、 访问者模式的应用场景
- 六、 总结
这篇文章带你详细认识一下设计模式中的访问者模式
一、 啥是访问者模式?
想象一下,你每年都要去医院做一次体检 🏥。 体检的项目有很多,比如量血压、验血、做心电图等等 🩺。 不同的医生负责不同的检查项目,比如内科医生负责量血压,化验师负责验血 🧪。
访问者模式,就是表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素类的前提下定义作用于这些元素的新操作!简单来说,就是把对对象的操作从对象本身分离出来,放到访问者中去执行 👨 ⚕️。
简单来说,就是让不同的“医生(访问者)”来“检查(访问)”你的“身体(元素)”,并执行不同的检查项目!👨 ⚕️+🏥
- 你想对一个对象结构中的对象执行不同的操作,但是你不想修改这些对象的类:就像你想让不同的医生检查你的身体,但是你不想修改你的身体结构 🧍!
- 你想在运行时动态地添加新的操作:就像你想临时增加一个检查项目,比如做核磁共振 🧲!
- 你想把操作和对象结构分离,降低耦合度:就像你想把检查项目和身体分离,让它们可以独立变化 🏥!
二、 为什么要用访问者模式?
用访问者模式,好处多多 👍:
- 分离操作:将操作和对象结构分离,降低耦合度 🔗!
- 增加新操作:可以在不修改对象结构的前提下,增加新的操作 ➕!
- 符合单一职责原则:每个访问者只负责一种操作 👨 ⚕️!
- 易于扩展:可以很容易地添加新的访问者和新的元素 🆕!
三、 访问者模式的实现方式
访问者模式主要包含以下几个角色:
- 访问者(Visitor):声明一个访问操作,针对对象结构中的每一种元素都提供一个访问方法 👨 ⚕️。
- 具体访问者(ConcreteVisitor):实现访问者声明的访问操作,执行具体的操作 👨 ⚕️。
- 元素(Element):定义一个接受访问操作的方法(accept),该方法接受一个访问者作为参数 🧍。
- 具体元素(ConcreteElement):实现元素定义的接受访问操作的方法,通常是将访问者传递给自身 🧍。
- 对象结构(ObjectStructure):包含一组元素,并提供一个方法来让访问者访问这些元素 🏥。
代码示例:
// 访问者接口:医生
public interface Doctor {
void visit(BodyPartA bodyPartA); // 检查身体部位 A
void visit(BodyPartB bodyPartB); // 检查身体部位 B
}
// 具体访问者:内科医生
public class InternalMedicineDoctor implements Doctor {
@Override
public void visit(BodyPartA bodyPartA) {
System.out.println("内科医生检查 " + bodyPartA.getName() + ":量血压");
}
@Override
public void visit(BodyPartB bodyPartB) {
System.out.println("内科医生检查 " + bodyPartB.getName() + ":听诊");
}
}
// 具体访问者:化验师
public class LabTechnician implements Doctor {
@Override
public void visit(BodyPartA bodyPartA) {
System.out.println("化验师检查 " + bodyPartA.getName() + ":验血");
}
@Override
public void visit(BodyPartB bodyPartB) {
System.out.println("化验师检查 " + bodyPartB.getName() + ":验尿");
}
}
// 元素接口:身体部位
public interface BodyPart {
String getName();
void accept(Doctor doctor); // 接受医生
}
// 具体元素:手臂
public class BodyPartA implements BodyPart {
@Override
public String getName() {
return "手臂";
}
@Override
public void accept(Doctor doctor) {
doctor.visit(this); // 接受医生,并将自身传递给医生
}
}
// 具体元素:心脏
public class BodyPartB implements BodyPart {
@Override
public String getName() {
return "心脏";
}
@Override
public void accept(Doctor doctor doctor) {
doctor.visit(this); // 接受医生,并将自身传递给医生
}
}
// 对象结构:身体
public class Body {
private List<BodyPart> bodyParts = new ArrayList<>();
public void add(BodyPart bodyPart) {
bodyParts.add(bodyPart);
}
public void remove(BodyPart bodyPart) {
bodyParts.remove(bodyPart);
}
public void accept(Doctor doctor) {
for (BodyPart bodyPart : bodyParts) {
bodyPart.accept(doctor); // 让每个身体部位接受医生
}
}
}
// 客户端
public class Client {
public static void main(String[] args) {
Body body = new Body(); // 创建身体对象
body.add(new BodyPartA()); // 添加手臂
body.add(new BodyPartB()); // 添加心脏
Doctor internalMedicineDoctor = new InternalMedicineDoctor(); // 创建内科医生
Doctor labTechnician = new LabTechnician(); // 创建化验师
body.accept(internalMedicineDoctor); // 让内科医生检查身体
body.accept(labTechnician); // 让化验师检查身体
}
}
分析:
Doctor
是访问者接口,定义了访问不同身体部位的visit
方法。InternalMedicineDoctor
和LabTechnician
是具体访问者,分别实现了内科医生和化验师的检查逻辑。BodyPart
是元素接口,定义了accept
方法,用于接受医生。BodyPartA
和BodyPartB
是具体元素,分别代表手臂和心脏,实现了accept
方法,将自身传递给医生。Body
是对象结构,包含了所有的身体部位,并提供了accept
方法,让每个身体部位接受医生。
输出结果:
内科医生检查 手臂:量血压
内科医生检查 心脏:听诊
化验师检查 手臂:验血
化验师检查 心脏:验尿
四、 访问者模式的优缺点
优点:
- 分离操作 🔗!
- 增加新操作 ➕!
- 符合单一职责原则 👨 ⚕️!
- 易于扩展 🆕!
缺点:
- 增加了系统的复杂度 😫!
- 违反了开闭原则(如果增加新的元素,需要修改所有的访问者) ❌!
- 元素必须暴露内部状态给访问者 🙈!
五、 访问者模式的应用场景
- 对对象结构中的对象执行不同的操作,但是你不想修改这些对象的类:就像你想让不同的医生检查你的身体,但是你不想修改你的身体结构 🧍!
- 需要在运行时动态地添加新的操作:就像你想临时增加一个检查项目,比如做核磁共振 🧲!
- 操作和对象结构之间的关系比较稳定:就像检查项目和身体之间的关系比较稳定 🏥!
六、 总结
- 访问者模式就像让不同的“医生(访问者)”来“检查(访问)”你的“身体(元素)”,并执行不同的检查项目! 👨 ⚕️+🏥
- 优点是分离操作、增加新操作、符合单一职责原则、易于扩展! 👍
- 缺点是增加复杂度、违反开闭原则、元素必须暴露内部状态! 👎
- 适用于需要对对象结构中的对象执行不同的操作,但是你不想修改这些对象的类的场景! 🎯
希望这个例子能让你彻底理解访问者模式! 💯 祝你学习愉快! 😄
热门推荐
海南旅游购票全攻略:如何选购性价比高的往返车票
金秋上海半日游攻略!4种主题带你逛遍经典打卡地
一文读懂:网络短视频侵权如何认定?“二次创作”是否构成侵权?
LCD1602引脚功能实战:一步步成为显示技术专家
精油按摩是什么?和芳疗、指压不同吗?7大优点、注意事项全介绍
计算大组合数模大数的算法详解
不同类型的风化简介
牛仔裤背后的经典故事
轻触自锁继电器:功能、参数与使用指南详解
最适合打游戏的电脑配置 爱打游戏的看过来!
过退休年龄人的误工费的法律依据是什么
以租代购销售合同:法律条款与实务操作指南
国际法:探究其重要性及在全球治理中的作用
教育心理学书籍推荐十佳
湛江水产养殖“游”向深远海:从传统渔排到智能化平台的转型升级
湛江海洋牧场步入养殖管理智能化时代,实现智慧监控全覆盖
千兆、2.5G和5G网口对比:如何选择适合的网络接口?
从麦克斯韦方程组开始,走进光的世界
戊戌变法后、辛亥革命前,唐才常领导起义、“讨贼勤王”
收紧!已不允许购买!多家银行出手了
如何改善居民的居住体验和生活品质?这些改善的措施和方向有哪些?
法庭程序出错的纠正机制及其法律实务探讨
法院送达程序违法怎么办
家庭如何关怀小腹胀痛的女性?
让传统醒狮舞出新“光”彩
想学PLC编程?先弄清5种PLC专用语言 !
葡萄酒标签上除了生产日期还应该标注哪些信息?
皮肤病血毒丸,警惕其潜在副作用与危害
项目时间进度怎么管理好
如何优化企业知识库目录结构,提高搜索引擎排名?