Python 设计模式:访问者模式

1. 什么是访问者模式?

访问者模式是一种行为设计模式,它允许你在不改变对象结构的前提下,定义新的操作。通过将操作封装在访问者对象中,访问者模式使得你可以在不修改元素类的情况下,向元素类添加新的功能。

访问者模式的核心思想是将操作与对象结构分离。通过引入一个访问者对象,允许你在不修改对象结构的情况下,定义新的操作。这样可以提高系统的灵活性和可扩展性。

访问者模式通常包含以下几个组成部分:

  • 访问者接口(Visitor Interface):定义访问者的接口,通常包含对每个元素类的访问方法。
  • 具体访问者(Concrete Visitor):实现访问者接口,定义对每个元素类的具体操作。
  • 元素接口(Element Interface):定义接受访问者的接口,通常包含一个接受访问者的方法。
  • 具体元素(Concrete Element):实现元素接口,定义具体的元素类。

访问者模式在软件设计中具有多种优点:

  • 分离操作与对象结构:通过将操作封装在访问者中,访问者模式使得操作与对象结构分离,增强了系统的灵活性。
  • 易于扩展:可以通过添加新的访问者来扩展系统的功能,而不需要修改现有的元素类。
  • 集中操作:所有操作都集中在访问者中,便于管理和维护。
# 访问者接口
class DiscountVisitor:def visit_book(self, book):passdef visit_electronic(self, electronic):pass# 具体访问者
class PercentageDiscount(DiscountVisitor):def visit_book(self, book):return book.price * 0.9  # 书籍享受10%的折扣def visit_electronic(self, electronic):return electronic.price * 0.85  # 电子产品享受15%的折扣# 元素接口
class Item:def accept(self, visitor):pass# 具体元素
class Book(Item):def __init__(self, price):self.price = pricedef accept(self, visitor):return visitor.visit_book(self)  # 接受访问者class Electronic(Item):def __init__(self, price):self.price = pricedef accept(self, visitor):return visitor.visit_electronic(self)  # 接受访问者# 客户端代码
if __name__ == "__main__":items = [Book(100), Electronic(200)]  # 创建购物车中的商品discount_visitor = PercentageDiscount()  # 创建折扣访问者for item in items:discounted_price = item.accept(discount_visitor)  # 计算折扣后的价格print(f"Discounted price: {discounted_price}")
  1. 访问者接口DiscountVisitor 类定义了访问者的接口,包含对每个元素类的访问方法(visit_bookvisit_electronic)。
  2. 具体访问者
    • PercentageDiscount 类实现了访问者接口,定义了对每个元素类的具体操作(计算折扣后的价格)。
    • visit_book 方法中,书籍享受10%的折扣;在 visit_electronic 方法中,电子产品享受15%的折扣。
  3. 元素接口Item 类定义了接受访问者的接口,包含一个 accept 方法。
  4. 具体元素
    • BookElectronic 类实现了元素接口,定义具体的商品类,并实现 accept 方法,接受访问者。
  5. 客户端代码
    • 在客户端代码中,创建购物车中的商品和折扣访问者,并通过调用 accept 方法来计算折扣后的价格。

2. 示例1:宠物店中的访问者模式

  • 访问者接口(Visitor Interface):定义访问者的接口,通常包含对每个宠物类的访问方法。
  • 具体访问者(Concrete Visitor):实现访问者接口,定义对每个宠物类的具体操作(如计算数量、平均体重和最大年龄)。
  • 元素接口(Element Interface):定义接受访问者的接口,通常包含一个接受访问者的方法。
  • 具体元素(Concrete Element):实现元素接口,定义具体的宠物类(如猫和狗)。
# 访问者接口
class PetVisitor:def visit_cat(self, cat):passdef visit_dog(self, dog):pass# 具体访问者:数量统计
class CountVisitor(PetVisitor):def __init__(self):self.cat_count = {'male': 0, 'female': 0}self.dog_count = {'male': 0, 'female': 0}def visit_cat(self, cat):self.cat_count[cat.gender] += 1def visit_dog(self, dog):self.dog_count[dog.gender] += 1def report(self):print(f"Cat Count: {self.cat_count['male']} males, {self.cat_count['female']} females")print(f"Dog Count: {self.dog_count['male']} males, {self.dog_count['female']} females")# 具体访问者:平均体重计算
class WeightVisitor(PetVisitor):def __init__(self):self.total_cat_weight = 0self.total_dog_weight = 0self.cat_count = 0self.dog_count = 0def visit_cat(self, cat):self.total_cat_weight += cat.weightself.cat_count += 1def visit_dog(self, dog):self.total_dog_weight += dog.weightself.dog_count += 1def report(self):avg_cat_weight = self.total_cat_weight / self.cat_count if self.cat_count > 0 else 0avg_dog_weight = self.total_dog_weight / self.dog_count if self.dog_count > 0 else 0print(f"Average Cat Weight: {avg_cat_weight:.2f} kg")print(f"Average Dog Weight: {avg_dog_weight:.2f} kg")# 具体访问者:最大年龄查找
class AgeVisitor(PetVisitor):def __init__(self):self.max_cat_age = 0self.max_dog_age = 0def visit_cat(self, cat):if cat.age > self.max_cat_age:self.max_cat_age = cat.agedef visit_dog(self, dog):if dog.age > self.max_dog_age:self.max_dog_age = dog.agedef report(self):print(f"Oldest Cat Age: {self.max_cat_age} years")print(f"Oldest Dog Age: {self.max_dog_age} years")# 具体元素:猫
class Cat:def __init__(self, gender, weight, age):self.gender = gender  # 'male' or 'female'self.weight = weight  # 体重self.age = age  # 年龄def accept(self, visitor):visitor.visit_cat(self)  # 接受访问者# 具体元素:狗
class Dog:def __init__(self, gender, weight, age):self.gender = gender  # 'male' or 'female'self.weight = weight  # 体重self.age = age  # 年龄def accept(self, visitor):visitor.visit_dog(self)  # 接受访问者# 客户端代码
if __name__ == "__main__":pets = [Cat('male', 4.5, 3),Cat('female', 3.0, 2),Dog('male', 10.0, 5),Dog('female', 8.5, 4),Cat('male', 5.0, 6),Dog('female', 9.0, 7)]count_visitor = CountVisitor()  # 创建数量统计访问者weight_visitor = WeightVisitor()  # 创建平均体重计算访问者age_visitor = AgeVisitor()  # 创建最大年龄查找访问者# 统计数量、计算平均体重和查找最大年龄for pet in pets:pet.accept(count_visitor)  # 统计数量pet.accept(weight_visitor)  # 计算平均体重pet.accept(age_visitor)  # 查找最大年龄# 输出统计结果count_visitor.report()weight_visitor.report()age_visitor.report()
Cat Count: 2 males, 1 females
Dog Count: 1 males, 2 females
Average Cat Weight: 4.17 kg
Average Dog Weight: 9.17 kg
Oldest Cat Age: 6 years
Oldest Dog Age: 7 years
  1. 访问者接口PetVisitor 类定义了访问者的接口,包含对每个宠物类的访问方法(visit_catvisit_dog)。

  2. 具体访问者

    • CountVisitor 类实现了访问者接口,负责统计宠物的数量(按性别分类)。
    • WeightVisitor 类实现了访问者接口,负责计算宠物的平均体重。
    • AgeVisitor 类实现了访问者接口,负责找出年龄最大的猫和狗。
  3. 具体元素

    • CatDog 类实现了元素接口,定义具体的宠物类,并实现 accept 方法,接受访问者。
  4. 客户端代码

    • 在客户端代码中,创建宠物对象(猫和狗)并创建数量统计、平均体重计算和最大年龄查找的访问者。
    • 通过调用 accept 方法,分别统计数量、计算平均体重和查找最大年龄,并打印结果。

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

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

相关文章

基于stm32的智能门锁系统

标题:基于stm32的智能门锁系统 内容:1.摘要 摘要:随着科技的飞速发展,人们对家居安全的要求日益提高,智能门锁系统应运而生。本研究的目的是设计并实现一个基于STM32的智能门锁系统。采用STM32微控制器作为核心控制单元,结合指纹…

GitHub 常见高频问题与解决方案(实用手册)

目录 1.Push 提示权限错误(Permission denied) 2.push 报错:rejected non-fast-forward 3.忘记添加 .gitignore,上传了无关文件 4. 撤销最近一次 commit 5.clone 太慢或失败 6.如何切换/创建分支 7.如何合并分支 8.如何删除远程分支 9.如何 Fork + PR(Pull Reque…

【MySQL数据库入门到精通-04 DML操作】

一、DML DML英文全称是Data Manipulation Language(数据操作语言),用来对数据库中表的数据记录进行增、删、改操作。 二、添加数据 1.给指定字段添加数据 代码如下(示例): insert into 表名 (字段1,字…

2022 年 9 月青少年软编等考 C 语言六级真题解析

目录 T1. 栈的基本操作T2. stack or queue思路分析T3. 合影效果T4. 发型糟糕的一天思路分析T1. 栈的基本操作 题目链接:SOJ D1188 此题为 2022 年 6 月三级第二题仅有栈操作的版本,见 2022 年 6 月青少年软编等考 C 语言三级真题解析中的 T2。 T2. stack or queue 题目链…

美创市场竞争力突出!《2025中国数据安全市场研究报告》发布

数据要素时代,数据已成国家战略性资源,数据安全关乎国家安全!数说安全发布的《2025中国数据安全市场研究报告》(以下简称《报告》)显示,2024年数据安全市场逆势增长,市场规模首次突破百亿。《报…

VUE Element-ui Message 消息提示组件自定义封装

为了让message 信息提示的更加方便快捷,减少不同地方的调用,避免代码的重复,特意再官方message 组件的基础上二次封装,使代码更加的优雅和高效。 实现效果: 代码组件: 封装成 message.js 文件,…

高防IP能抵御哪些类型的网络攻击?

高防IP(High Defense IP)是一种专门针对网络攻击设计的防护服务,主要通过流量清洗、协议分析、行为检测等技术抵御多种网络攻击。以下是其能防御的主要攻击类型及原理: ​​一、常见防御的攻击类型​​ ​​DDoS攻击(分…

小红书文字配图平替工具

小红书的文字配图只有手机版有,想找一个电脑版的,查了一下。以下是几款类似小红书风格的花字、艺术字生成工具,适合制作吸睛的社交媒体配图,分为 手机APP 和 在线工具 两类,供你选择: 一、手机APP推荐 醒图…

【浙江大学DeepSeek公开课】走向数字社会:从DeepSeek到群体智慧

从DeepSeek到群体智慧 一、人工智能发展脉络二、DeepSeek大模型的意义与特点三、人工智能促进社会数字化转型四、群体智慧与数字社会 一、人工智能发展脉络 图灵与图灵机:1937年,图灵发表论文《On computable numbers, with an application to the Ents…

解读大型语言模型:从Transformer架构到模型量化技术

一、生成式人工智能概述 生成式人工智能(Generative Artificial Intelligence)是一种先进的技术,能够生成多种类型的内容,包括文本、图像、音频以及合成数据等。其用户界面的便捷性极大地推动了其广泛应用,用户仅需在…

JSON实现动态按钮管理的Python应用

在开发桌面应用程序时,动态生成用户界面元素并根据配置文件灵活管理是一项常见需求。本文将介绍如何使用Python的wxPython库结合JSON配置文件,开发一个支持动态按钮创建、文件执行和配置管理的桌面应用程序。该应用允许用户通过设置界面配置按钮名称和关…

序章:写在前面

目录 为什么要学习 Python?那么,Python 到底是什么呢?Python 的用户多吗?Python 的语法究竟是怎样的?C 语言JavaPython Python 好学吗? 为什么要学习 Python? 这个问题或许会让不少人感到不解。…

onlyoffice历史版本功能实现,版本恢复功能,编辑器功能实现 springboot+vue2

文章目录 oonlyoffice历史版本功能实现 (编辑器功能实现)springbootvue2前提 需要注意把这个 (改成自己服务器的ip或者域名) 改成 自己服务器的域名或者地址1. onloyoffice 服务器部署 搜索其他文章2. 前段代码 vue 22.1 需要注意把这个 (改成自己服务器…

解决ubuntu server修改为中文后乱码问题(改回英文)

操作步骤 1.安装英文语言包 sudo apt-get install language-pack-en2.编辑/etc/default/locale文件 sudo vim /etc/default/locale修改为以下内容: LANG"en_US.UTF-8" LANGUAGE"en_US:en" LC_ALL"en_US.UTF-8"3.应用配置 sudo l…

安卓的Launcher 在哪个环节进行启动

安卓Launcher在系统启动过程中的关键环节启动,具体如下: 内核启动:安卓设备开机后,首先由引导加载程序启动Linux内核。内核负责初始化硬件设备、建立内存管理机制、启动系统进程等基础工作,为整个系统的运行提供底层支…

数据通信学习笔记之OSPF其他内容2

OSPF 与 BFD 联动 网络上的链路故障或拓扑变化都会导致设备重新进行路由计算,所以缩短路由协议的收敛时间对于提高网络的性能是非常重要的。 OSPF 与 BFD 联动就是将 BFD 和 OSPF 关联起来,一旦与邻居之间的链路出现故障,BFD 对完品以&…

数据库原理及应用mysql版陈业斌实验四

🏝️专栏:Mysql_猫咪-9527的博客-CSDN博客 🌅主页:猫咪-9527-CSDN博客 “欲穷千里目,更上一层楼。会当凌绝顶,一览众山小。” 目录 实验四索引与视图 1.实验数据如下 student 表(学生表&…

[密码学实战]密评考试训练系统v1.0程序及密评参考题库(获取路径在文末)

[密码学实战]密评考试训练系统v1.0程序及密评参考题库 引言:密评考试的重要性与挑战 商用密码应用安全性评估(简称"密评") 作为我国密码领域的重要认证体系,已成为信息安全从业者的必备技能。根据国家密码管理局最新数据,截至2024年6月,全国仅有3000余人持有…

蓝桥杯练习题2

动态规划 动态规划三大题型:计数问题、最值问题、存在性问题; 【最小权值】-- 最值问题 【题目分析】 import java.util.Arrays; Arrays类中的一个方法:Arrays.fill(int[] m,int n) //给 int 类型(或者char类型/Long类型...)的数组全部空间…

【集群IP管理分配技术_DHCP】二、DHCP核心功能与技术实现

一、智能 IP 地址分配功能与技术实现​ 1.1 功能概述​ 智能 IP 地址分配是 DHCP 中间件的核心功能之一,它打破了传统 DHCP 固定的分配模式,能够根据网络的实时状态、客户端类型、接入位置等多种因素,动态且合理地分配 IP 地址。例如&#…