【设计模式-享元】

ops/2024/9/23 23:12:27/

Flyweight Pattern(享元模式) 是一种结构型设计模式,旨在通过共享对象来减少内存使用和提高性能。享元模式特别适用于需要大量相似对象的场景,可以有效地减少内存开销。

核心思想

享元模式通过将对象的共享部分(共享状态)和对象的独立部分(非共享状态)分开,使得可以重用已存在的对象。通过共享相同的对象实例,显著减少内存消耗,从而提高应用程序的性能。

组成部分

  • Flyweight(享元接口): 声明了操作的接口,通常包含共享状态的操作。

  • ConcreteFlyweight(具体享元类): 实现了享元接口,维护共享状态。具体享元对象是可共享的。

  • UnsharedConcreteFlyweight(非共享享元类): 如果有一些状态不能共享(即特定于某个对象的状态),则可以将其定义在非共享享元类中。

  • FlyweightFactory(享元工厂): 负责创建和管理享元对象。它确保当请求一个对象时,返回的是已经存在的对象,而不是创建新的对象。

UML图

在这里插入图片描述

Java代码

以下是一个简单的 Java 示例,展示了如何使用享元模式。假设我们在一个文本编辑器中,每个字符都可能具有某种样式(如字体、颜色等),我们希望重用相同的字符对象以节省内存。

java">import java.util.HashMap;
import java.util.Map;// Flyweight
interface Character {void display(int size);
}// ConcreteFlyweight
class ConcreteCharacter implements Character {private final char symbol;public ConcreteCharacter(char symbol) {this.symbol = symbol;}@Overridepublic void display(int size) {System.out.println("Character: " + symbol + ", Size: " + size);}
}// FlyweightFactory
class CharacterFactory {private final Map<Character, Character> characters = new HashMap<>();public Character getCharacter(char symbol) {Character character = characters.get(new ConcreteCharacter(symbol));if (character == null) {character = new ConcreteCharacter(symbol);characters.put(character, character);}return character;}
}// Client
public class FlyweightPatternExample {public static void main(String[] args) {CharacterFactory factory = new CharacterFactory();// 共享字符Character a = factory.getCharacter('A');Character b = factory.getCharacter('B');// 使用不同的大小显示相同的字符a.display(12);a.display(10);b.display(14);// 进一步共享Character a2 = factory.getCharacter('A');System.out.println("Are a and a2 the same? " + (a == a2)); // true}
}

解释

  • Character(享元接口): 定义了 display 方法,用于显示字符和其大小。

  • ConcreteCharacter(具体享元类): 实现了 Character 接口,维护了字符的共享状态(即字符本身)。

  • CharacterFactory(享元工厂): 负责管理和共享字符对象,确保每个字符只有一个实例。

  • Client: 客户端使用工厂获取字符对象,并可以多次使用相同的字符对象来节省内存。

优点

  • 内存节省: 通过共享相同的对象实例,显著减少内存占用。

  • 提高性能: 减少了对象创建和垃圾回收的开销,提高了性能。

  • 灵活性: 允许使用不同的组合来创建复杂的对象,增强了系统的灵活性。

缺点

  • 复杂性增加: 引入了额外的工厂和共享逻辑,使系统设计更加复杂。

  • 管理共享状态: 需要小心管理共享状态,避免线程不安全和数据不一致的问题。

使用场景

  • 文本处理: 在文档编辑器中,字符和样式可以共享,减少内存使用。
  • 图形系统: 在图形应用程序中,许多图形元素(如树、花、建筑物等)可能有相同的外观,但不同的位置。
  • 游戏开发: 游戏中的角色、道具等对象可以使用享元模式进行共享。

总结

享元模式通过共享对象来减少内存消耗,提高性能,尤其适合需要大量相似对象的场景。它通过合理管理共享和非共享状态,优化资源使用,增强系统的灵活性和性能。尽管引入了额外的复杂性,但在合适的应用场景中,享元模式能够带来显著的好处。


http://www.ppmy.cn/ops/115008.html

相关文章

6、论文阅读:水下图像增强基准数据集及其他数据集

水下图像增强基准数据集及其他数据集 前言引言贡献现有方法、评估指标和数据集:概述水下增强方法基于补充信息的方法基于非物理模型的方法基于物理模型的方法数据驱动的方法水下图像质量评估完整参考指标没有参考指标水下图像数据集提出基准数据集数据收集参考图像的生成评估与…

尚品汇-Jenkins部署构建服务模块、Linux快照备份(五十七)

目录&#xff1a; &#xff08;1&#xff09;构建作业&#xff08;server-gateway&#xff09; &#xff08;2&#xff09;构建service_product模块 &#xff08;3&#xff09;演示添加新代码 &#xff08;4&#xff09;学会使用linux快照 &#xff08;1&#xff09;构建作…

Android 后台服务之Persistent 属性

在 Android 开发中,有时我们需要后台服务持续运行,以保持应用的某些功能。例如,音乐播放器需要在后台播放音乐,或者健康应用需要持续跟踪用户的运动数据。后台服务是 Android 中的一种组件,它不与用户界面交互,能够在后台执行长时间运行的任务。由于 Android 系统的资源管…

微信抢红包设计

包几个红包&#xff0c;发红包的总金额塞钱进红包&#xff0c;弹出支付的界面&#xff0c;支付完之后就会被发到 群里抢拆红包&#xff0c;才是真正的抢红包 红包金额怎么分配 每个人至少抢到1分钱人数抢到的金额之和就应该是红包的总金额一个人抢的红包不应该极度的大&#…

【Python报错已解决】AttributeError: ‘WindowsPath‘ object has no attribute ‘rstrip‘

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 专栏介绍 在软件开发和日常使用中&#xff0c;BUG是不可避免的。本专栏致力于为广大开发者和技术爱好者提供一个关于BUG解决的经…

快来看看!如何利用工具轻松实现自动加好友

大家都知道&#xff0c;手动添加大量好友不仅耗时&#xff0c;还很容易出错。这时候&#xff0c;借助一些工具就显得尤为重要&#xff01; 今天&#xff0c;就给大家分享一个多微管理工具&#xff0c;让你实现自动加好友。 首先&#xff0c;在多微管理工具上登录你的微信账号…

清空当前机器所有Docker容器和镜像

sudo docker stop $(sudo docker ps -aq) sudo docker rm $(sudo docker ps -aq) sudo docker rmi $(sudo docker images -q)删除当前机器上的所有Docker镜像是一个高风险操作&#xff0c;因为它会删除所有镜像&#xff0c;包括那些可能正在被容器使用的镜像。在执行此操作之前…

【Verilog学习日常】—牛客网刷题—Verilog快速入门—VL24

边沿检测 有一个缓慢变化的1bit信号a&#xff0c;编写一个程序检测a信号的上升沿给出指示信号rise&#xff0c;当a信号出现下降沿时给出指示信号down。 注&#xff1a;rise,down应为单脉冲信号&#xff0c;在相应边沿出现时的下一个时钟为高&#xff0c;之后恢复到0&#xff0…