当前位置: 首页 > news >正文

Python 设计模式:享元模式

1. 什么是享元模式?

享元模式是一种结构型设计模式,旨在通过共享对象来减少内存使用和提高性能。它特别适用于需要大量相似对象的场景,通过共享相同的对象来避免重复创建,从而节省内存和提高效率。

享元模式的核心思想是将对象的状态分为两部分:

  1. 内部状态(Intrinsic State):对象的共享部分,通常是不可变的,多个对象可以共享这些状态。
  2. 外部状态(Extrinsic State):对象的非共享部分,通常是可变的,依赖于具体的上下文或环境。

享元模式通常包含以下几个组成部分:

  • 享元接口(Flyweight Interface):定义享元对象的接口,通常包含一个方法来接受外部状态。
  • 具体享元(Concrete Flyweight):实现享元接口,存储内部状态,并可以接受外部状态。
  • 享元工厂(Flyweight Factory):负责创建和管理享元对象,确保共享相同的享元实例。
class Shape:def draw(self, color):passclass Circle(Shape):def __init__(self, radius):self.radius = radius  # 内部状态(共享部分)def draw(self, color):print(f"Drawing a {color} circle with radius {self.radius}")class ShapeFactory:def __init__(self):self.shapes = {}  # 存储共享的圆形对象def get_circle(self, radius):if radius not in self.shapes:self.shapes[radius] = Circle(radius)  # 创建新的圆形对象return self.shapes[radius]  # 返回共享的圆形对象if __name__ == "__main__":factory = ShapeFactory()# 创建并共享相同半径的圆circle1 = factory.get_circle(5)circle1.draw("red")  # 外部状态(颜色)circle2 = factory.get_circle(5)circle2.draw("blue")  # 外部状态(颜色)circle3 = factory.get_circle(10)circle3.draw("green")  # 外部状态(颜色)# 验证 circle1 和 circle2 是同一个对象print(f"circle1 is circle2: {circle1 is circle2}")  # 输出: True
  • 内部状态(Intrinsic State):在这个示例中,Circle 类的 radius 属性是内部状态,它是共享的,多个 Circle 对象可以共享相同的半径。
  • 外部状态(Extrinsic State):在 draw 方法中,颜色参数是外部状态,它是可变的,依赖于具体的上下文或环境。
  1. 享元接口Shape 类定义了享元对象的接口,包含一个 draw 方法。

  2. 具体享元类

    • Circle 类实现了 Shape 接口,存储内部状态(半径)并实现 draw 方法。
    • radius 是内部状态(Intrinsic State),它是共享的,多个 Circle 对象可以共享相同的半径。
  3. 享元工厂

    • ShapeFactory 类负责创建和管理享元对象。它使用一个字典来存储已经创建的圆形对象,确保相同半径的圆形对象被共享。
    • 当请求一个圆形时,如果该半径的圆形已经存在,则返回共享的对象;否则,创建一个新的对象并存储。
  4. 客户端代码

    • 在客户端代码中,创建 ShapeFactory 实例并请求圆形对象。
    • circle1circle2 共享相同的半径(5),但可以使用不同的颜色(外部状态)进行绘制。
    • 通过 circle1 is circle2 的比较,可以验证这两个对象实际上是同一个对象。

享元模式在软件设计中具有多种优点:

  • 节省内存:通过共享对象,减少了内存的使用。对于大量相似对象的场景,享元模式可以显著降低内存占用。

ShapeFactory 类负责管理 Circle 对象的创建。每当请求一个圆形时,如果该半径的圆形已经存在,则返回共享的对象;否则,创建一个新的对象并存储。通过这种方式,如果多个圆形对象具有相同的半径,它们将共享同一个 Circle 实例,而不是为每个对象创建一个新的实例。这显著减少了内存的使用。

class ShapeFactory:def __init__(self):self.shapes = {}  # 存储共享的圆形对象def get_circle(self, radius):if radius not in self.shapes:self.shapes[radius] = Circle(radius)  # 创建新的圆形对象return self.shapes[radius]  # 返回共享的圆形对象
  • 提高性能:减少了对象的创建和销毁,提高了性能。共享对象的使用可以减少系统的负担,提升响应速度。

创建和管理 Circle 对象的逻辑被集中在 ShapeFactory 中。由于相同半径的圆形对象只会被创建一次,后续的请求将直接返回已存在的对象。这里,circle1circle2 实际上是同一个对象。通过避免重复创建相同的对象,享元模式减少了系统的负担,从而提高了性能。

circle1 = factory.get_circle(5)
circle2 = factory.get_circle(5)
  • 灵活性:可以根据需要动态地添加新的享元对象,而不影响现有对象的状态。

ShapeFactory 中,新的 Circle 对象可以根据需要被创建并存储。每当请求一个新的半径时,工厂会检查是否已经存在相应的对象。这种设计允许系统在运行时灵活地添加新的享元对象,而不需要修改现有的对象或逻辑。这种灵活性使得系统能够适应变化的需求。

def get_circle(self, radius):if radius not in self.shapes:self.shapes[radius] = Circle(radius)  # 创建新的圆形对象return self.shapes[radius]  # 返回共享的圆形对象

2. 示例1:音频播放器中的享元模式

# 享元接口
class Audio:def play(self):pass# 具体享元类
class MP3(Audio):def __init__(self, file_path):self.file_path = file_path  # 内部状态(共享部分)def play(self):print(f"Playing audio from: {self.file_path}")# 享元工厂
class AudioFactory:def __init__(self):self.audios = {}  # 存储共享的音频对象def get_audio(self, file_path):if file_path not in self.audios:self.audios[file_path] = MP3(file_path)  # 创建新的音频对象return self.audios[file_path]  # 返回共享的音频对象# 客户端代码
if __name__ == "__main__":factory = AudioFactory()# 创建并共享相同文件路径的音频对象audio1 = factory.get_audio("song1.mp3")audio1.play()  # 播放音频audio2 = factory.get_audio("song1.mp3")audio2.play()  # 播放相同的音频audio3 = factory.get_audio("song2.mp3")audio3.play()  # 播放不同的音频# 验证 audio1 和 audio2 是同一个对象print(f"audio1 is audio2: {audio1 is audio2}")  # 输出: True
Playing audio from: song1.mp3
Playing audio from: song1.mp3
Playing audio from: song2.mp3
audio1 is audio2: True
  1. 享元接口Audio 类定义了音频对象的接口,包含一个 play 方法。

  2. 具体享元类

    • MP3 类实现了 Audio 接口,存储内部状态(文件路径)并实现 play 方法。
    • file_path 是内部状态(Intrinsic State),它是共享的,多个 MP3 对象可以共享相同的文件路径。
  3. 享元工厂

    • AudioFactory 类负责创建和管理享元对象。它使用一个字典来存储已经创建的音频对象,确保相同文件路径的音频对象被共享。
    • 当请求一个音频时,如果该文件路径的音频已经存在,则返回共享的对象;否则,创建一个新的对象并存储。
  4. 客户端代码

    • 在客户端代码中,创建 AudioFactory 实例并请求音频对象。
    • audio1audio2 共享相同的文件路径(song1.mp3),但可以独立播放。
    • 通过 audio1 is audio2 的比较,可以验证这两个对象实际上是同一个对象。

相关文章:

  • XPath 语法入门
  • Vue 实例 VM 访问属性
  • 不可变数据:基于持久化数据结构的状态管理
  • 初级云计算运维工程师学习二
  • 110. 平衡二叉树
  • MYSQL—两阶段提交
  • 影刀RPA怎么和AI结合,制作自动采集小红书爆款文章+自动用AI改写标题、内容+用AI文生图生成发文图片+自动在小红书上发布文章
  • 【NLP】This Post Is All You Need阅读笔记
  • 【数字图像处理】立体视觉信息提取
  • Relay IR的核心数据结构
  • Docker 与 Docker-Compose 的区别
  • leetcode day36 01背包问题 494
  • 08_Docker Portainer可视化管理
  • 【Linux】47.高级IO(1)
  • SQLiteDatabase 增删改查(CRUD)详细操作
  • Java函数生成实际应用案例:数据处理流水线
  • # 基于PyTorch的食品图像分类系统:从训练到部署全流程指南
  • 基于javaweb的SpringBoot校园失物招领系统设计与实现(源码+文档+部署讲解)
  • 鸿蒙NEXT开发权限工具类(申请授权相关)(ArkTs)
  • Python-27:游戏英雄升级潜力评估
  • 乌克兰关切有中国人在俄军中服务,外交部:坚决反对无端指责
  • 特朗普特使将赴俄见普京,俄方:美俄间谈判艰难且耗时
  • 打造“旧书朋友圈”,“淘书乐”为旧书找“新朋友”
  • 智飞生物一季度营收下滑79%,连续三个季度亏损,称业绩波动与行业整体趋势一致
  • 体坛联播|利兹联、伯恩利重返英超,北京淘汰北控队晋级四强
  • 商务部:中国加快推进服务业扩大开放综合试点为世界注入更多确定性