【设计模式】享元模式(Flyweight Pattern)

server/2024/12/15 6:27:33/

享元模式(Flyweight Pattern)是一种结构型设计模式,它通过共享尽可能多的对象来有效支持大量细粒度的对象。这个模式主要用于减少内存使用和提高性能,特别是在需要创建大量相似对象的场景中。享元模式的核心思想是将对象的状态分为内在状态(Intrinsic State)和外在状态(Extrinsic State),其中内在状态存储在对象内部,并且对于所有引用该对象的地方都是相同的;而外在状态则由客户端管理并传递给对象。

享元模式的关键要素

  1. Flyweight (享元)
    这是一个接口或抽象类,声明了具体享元类应该实现的操作。
    它定义了哪些方法可以访问内在状态,以及如何接收外在状态。
    java
    深色版本
    public interface Flyweight {
    void operation(int extrinsicState);
    }

  2. ConcreteFlyweight (具体享元)
    实现 Flyweight 接口的具体类。
    包含内在状态,并且可能会根据传入的外在状态执行不同的操作。
    java
    深色版本
    public class ConcreteFlyweight implements Flyweight {
    private final String intrinsicState;

    public ConcreteFlyweight(String intrinsicState) {
    this.intrinsicState = intrinsicState;
    }

    @Override
    public void operation(int extrinsicState) {
    System.out.println("Intrinsic State = " + this.intrinsicState);
    System.out.println("Extrinsic State = " + extrinsicState);
    }
    }

  3. UnsharedConcreteFlyweight (非共享具体享元)
    如果某些对象不能被共享,或者它们包含的全部是外在状态,则可以创建非共享的具体享元类。
    这些对象不会被池化管理,而是直接由客户端创建和销毁。
    java
    深色版本
    public class UnsharedConcreteFlyweight implements Flyweight {
    @Override
    public void operation(int extrinsicState) {
    // 操作逻辑
    }
    }

  4. FlyweightFactory (享元工厂)
    负责管理和创建享元对象。
    它通常维护一个享元对象的缓存(如 HashMap),以确保相同内在状态的对象只创建一次,并能在后续请求时返回已存在的实例。
    java
    深色版本
    import java.util.HashMap;
    import java.util.Map;

public class FlyweightFactory {
private final Map<String, Flyweight> flyweights = new HashMap<>();

public Flyweight getFlyweight(String key) {return flyweights.computeIfAbsent(key, k -> new ConcreteFlyweight(k));
}

}
使用享元模式的好处
节省内存:通过共享不可变的部分,减少了内存占用。
提升性能:当需要频繁创建和销毁大量相似对象时,享元模式可以通过重用现有对象来加快程序运行速度。
简化对象管理:客户端只需要关心外在状态,而不需要处理每个对象的所有细节。
示例应用场景
文本编辑器中的字符格式:在一个大型文档中,如果每个字符都单独保存其字体、颜色等属性信息,将会消耗大量的内存。使用享元模式,可以将这些格式信息作为内在状态进行共享,而将字符位置等信息作为外在状态由客户端管理。
GUI 系统中的图标:对于 GUI 应用程序,可能有多个组件使用相同的图标。通过享元模式,可以确保每个唯一的图标只加载一次,并且可以在不同地方重复使用。
游戏开发中的敌人角色:在一些游戏中,同一类型的敌人角色可能具有相同的外观和行为。使用享元模式,可以将这些共同特征作为内在状态,而将位置、健康值等变化的数据作为外在状态。
注意事项
内外状态分离:正确区分内在状态和外在状态是成功应用享元模式的关键。内在状态应该是不可变的,而外在状态则是可变的并且由客户端提供。
线程安全问题:如果享元对象将在多线程环境中使用,必须确保内在状态的线程安全性。由于内在状态是不可变的,因此这通常不是问题,但仍然需要注意任何可能影响到对象共享的行为。
总之,享元模式适用于那些需要高效地管理大量相似对象的应用程序。通过合理利用对象共享机制,它可以显著降低系统资源消耗,同时保持良好的性能表现。


http://www.ppmy.cn/server/150280.html

相关文章

redis集群安装部署 redis三主三从集群

redis集群安装部署 redis三主三从集群 1、下载redis2、安装redis集群 三主三从3、配置redis开机自启动3.1、建立启动脚本3.2、复制多份redis启动脚本给集群使用3.3、添加可执行权限3.4、配置开机自启动 1、下载redis 本次redis安装部署选择当前最新的稳定版本7.4.1 下载链接: …

【ESP32】Arduino开发 | ADC模数转换器 + 单次转换例程

关于ESP32的ADC外设的详细介绍放在了ESP-IDF开发的对应文章中&#xff0c;通过跳转链接可以找到。 1. API 1.1 读取ADC&#xff08;原始值&#xff09; uint16_t analogRead(uint8_t pin) pin&#xff1a;ADC管脚。 1.2 读取ADC&#xff08;电压值&#xff09; uint32_t a…

日拱一卒(17)——leetcode学习记录:定长字符串中元音字符的数量

一、题目 给你字符串 s 和整数 k 。 请返回字符串 s 中长度为 k 的单个子字符串中可能包含的最大元音字母数。 英文中的 元音字母 为&#xff08;a, e, i, o, u&#xff09;。 示例 1&#xff1a; 输入&#xff1a;s "abciiidef", k 3 输出&#xff1a;3 解释&…

概率、似然、最小二乘

参考文章 通过例子理解最大似然估计 最大似然估计和最小二乘估计的区别与联系 SLAM的发展以及分类 移动机器人运动模型 深入理解最大似然估计与最大后验估计&#xff1a;概率中的两大估计法 从贝叶斯公式到卡尔曼滤波 Kalman滤波通俗理解实际应用 简单总结 概率&#xff1a;…

Visual Studio Community安装WebForm创建和数据连接

目录 一、安装Visual Studio Community 1.下载Visual Studio Community 2.双击安装 3.配置 二、项目创建 1.创建新项目 2.配置新项目 3.创建web窗体 4.代码分部 5.文件打开问题 三、数据库访问 1.自己先装一个sql server 2.创建实体数据模型 3.创建连接 4.修改更…

react-dnd 拖拽事件与输入框的文本选中冲突

问题描述 当我们使用拖拽库的时候&#xff0c;往往会遇到拖拽的一个元素他的子孙元素有输入框类型的dom节点&#xff0c;当拖拽的事件绑定在该元素身上时候&#xff0c;发现子孙的输入框不能进行文本选中了&#xff0c;会按住鼠标去选中文本的时候会触发拖拽 实际的效果&…

python制造一个报错

在Python中&#xff0c;制造一个报错&#xff08;异常&#xff09;可以通过多种方式实现。最常见的方法之一是直接引发一个异常。以下是一些例子&#xff0c;展示了如何制造不同类型的报错&#xff1a; 引发ValueError异常&#xff1a; try:# 制造一个 ValueError 异常raise Va…

PTL系统助力新能源汽车生产线的物料精准管理

在新能源汽车生产线蓬勃发展的当下&#xff0c;多车型混合生产模式已成为行业常态。传统的物料管理方式在面对多种型号新能源汽车生产时&#xff0c;由于缺乏精准的指引和高效的防错机制&#xff0c;常常出现物料错拿导致产品不良&#xff0c;以及因核对物料耗费大量时间而造成…