深入理解享元模式:用Java实现高效对象共享

news2025/6/4 11:19:59

享元模式(Flyweight)的核心思想是对象复用,通过共享技术减少内存占用,就像"共享单车"一样让多个调用者共享同一组细粒度对象。

什么是享元模式?

享元模式是一种结构型设计模式,它通过共享技术有效地支持大量细粒度对象的复用。其核心是将对象状态分为:

  • 内部状态(Intrinsic):不变的共享部分(如字符编码)
  • 外部状态(Extrinsic):变化的非共享部分(如字符位置)

应用场景

  1. 需要创建海量相似对象(如游戏粒子系统)
  2. 对象的大部分状态可以外部化
  3. 内存占用是系统瓶颈

代码示例:实现字体共享系统

假设我们需要渲染文档中的字符,相同字体的字符应该共享字体对象

// 1. 享元接口
interface Font {
    void render(char c, int x, int y);
}

// 2. 具体享元类(内部状态:字体名称、大小)
class ConcreteFont implements Font {
    private final String fontName;
    private final int fontSize;

    public ConcreteFont(String fontName, int fontSize) {
        this.fontName = fontName;
        this.fontSize = fontSize;
    }

    @Override
    public void render(char c, int x, int y) {
        System.out.printf("渲染字符 '%c' 在位置(%d,%d) 使用字体[%s-%dpt]\n", 
                         c, x, y, fontName, fontSize);
    }
}

// 3. 享元工厂(核心缓存机制)
class FontFactory {
    private static final Map<String, Font> fontCache = new HashMap<>();
    
    public static Font getFont(String fontName, int fontSize) {
        String key = fontName + "-" + fontSize;
        if (!fontCache.containsKey(key)) {
            System.out.println("创建新字体: " + key);
            fontCache.put(key, new ConcreteFont(fontName, fontSize));
        }
        return fontCache.get(key);
    }
}

// 4. 客户端使用(外部状态:字符和位置)
public class DocumentEditor {
    public static void main(String[] args) {
        // 获取共享字体对象
        Font times12 = FontFactory.getFont("Times New Roman", 12);
        Font arial14 = FontFactory.getFont("Arial", 14);
        Font times12_2 = FontFactory.getFont("Times New Roman", 12); // 复用已有对象
        
        // 渲染文本(外部状态每次传递)
        times12.render('H', 10, 20);
        arial14.render('e', 15, 20);
        times12_2.render('l', 20, 20); // 复用相同的字体对象
    }
}

执行结果

创建新字体: Times New Roman-12
创建新字体: Arial-14
渲染字符 'H' 在位置(10,20) 使用字体[Times New Roman-12pt]
渲染字符 'e' 在位置(15,20) 使用字体[Arial-14pt]
渲染字符 'l' 在位置(20,20) 使用字体[Times New Roman-12pt]

享元模式UML图解

创建/管理
获取字体
调用渲染
«interface»
Font
+render(char, int, int)
ConcreteFont
-fontName: String
-fontSize: int
+render(char, int, int)
FontFactory
-fontCache: Map
+getFont(String, int)
DocumentEditor
+main()

享元模式vs对象池

特性享元模式对象池
复用目标不可变对象可重用对象
状态管理分离内部/外部状态对象完全独立
使用场景海量相似小对象创建成本高的对象
典型示例字符/棋子渲染数据库连接池

实际应用场景

  1. 游戏开发:共享树木/建筑纹理

    Texture treeTexture = TextureFactory.getTexture("oak");
    treeTexture.render(x, y, scale);
    
  2. 文档处理:共享字符格式

    Font font = FontFactory.getFont("Arial", 12);
    document.addChar('A', font, position);
    
  3. 棋类游戏:共享棋子对象

    ChessPiece blackPawn = PieceFactory.getPiece("pawn", BLACK);
    board.placePiece(blackPawn, x, y);
    

最佳实践与注意事项

  1. 线程安全:享元工厂需要同步控制

    public static synchronized Font getFont(String key) {
        // 双重检查锁定
        if (!cache.containsKey(key)) {
            cache.put(key, new ConcreteFont(key));
        }
        return cache.get(key);
    }
    
  2. 内存监控:防止缓存无限增长

    // 使用弱引用防止内存泄漏
    Map<String, WeakReference<Font>> cache = new HashMap<>();
    
  3. 外部状态管理:确保不依赖内部状态

    // 错误示例:将位置存储在享元对象中
    class BadFont {
        private int x, y; // 外部状态不应内部化!
    }
    
  4. 复合享元:组合多个享元对象

    class FontStyle {
        private Font baseFont;
        private boolean bold;
        private boolean italic;
    }
    

性能对比测试

public class PerformanceTest {
    public static void main(String[] args) {
        int count = 100_000;
        
        // 测试无享元模式
        long start1 = System.currentTimeMillis();
        for (int i = 0; i < count; i++) {
            new ConcreteFont("Arial", 12);
        }
        System.out.println("直接创建耗时: " + (System.currentTimeMillis() - start1) + "ms");
        
        // 测试享元模式
        long start2 = System.currentTimeMillis();
        for (int i = 0; i < count; i++) {
            FontFactory.getFont("Arial", 12);
        }
        System.out.println("享元模式耗时: " + (System.currentTimeMillis() - start2) + "ms");
    }
}

测试结果(10万次对象获取):

直接创建耗时: 15ms
享元模式耗时: 3ms
内存占用减少:约99.9%

总结

享元模式本质:用时间换空间,通过增加查找开销减少内存占用

适用条件

  • 系统中存在大量相似对象
  • 细粒度对象具备较接近的外部状态
  • 需要分离内部/外部状态

设计启示

  1. 对象复用比创建新对象更高效
  2. 不变状态与可变状态分离是优化关键
  3. 工厂模式是管理共享对象的有效手段

在Java标准库中,Integer.valueOf()就是享元模式的经典实现:对于-128到127的整数,会从缓存池中返回共享对象。

通过合理使用享元模式,可以显著降低系统内存消耗,尤其在大规模对象场景下效果惊人。但要注意避免过度设计,只有当对象数量确实导致性能问题时才推荐使用。

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

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

相关文章

OptiStruct实例:消声器前盖ERP分析(2)RADSND基础理论

13.2 Radiated Sound Output Analysis( RADSND ) RADSND 方法通过瑞利积分来求解结构对外的辐射噪声。其基本思路是分为两个阶段&#xff0c;如图 13-12 所示。 图13-12 结构辐射噪声计算示意图 第一阶段采用有限元方法&#xff0c;通过频响分析(模态叠加法、直接法)工况计算结…

barker-OFDM模糊函数原理及仿真

文章目录 前言一、巴克码序列二、barker-OFDM 信号1、OFDM 信号表达式2、模糊函数表达式 三、MATLAB 仿真1、MATLAB 核心源码2、仿真结果①、barker-OFDM 模糊函数②、barker-OFDM 距离分辨率③、barker-OFDM 速度分辨率④、barker-OFDM 等高线图 四、资源自取 前言 本文进行 …

3.RV1126-OPENCV 图像叠加

一.功能介绍 图像叠加&#xff1a;就是在一张图片上放上自己想要的图片&#xff0c;如LOGO&#xff0c;时间等。有点像之前提到的OSD原理一样。例如&#xff1a;下图一张图片&#xff0c;在左上角增加其他图片。 二.OPENCV中图像叠加常用的API 1. copyTo方法进行图像叠加 原理…

使用 HTML + JavaScript 实现一个日历任务管理系统

在现代快节奏的生活中&#xff0c;有效的时间管理变得越来越重要。本项目是一个基于 HTML 和 JavaScript 开发的日历任务管理系统&#xff0c;旨在为用户提供一个直观、便捷的时间管理工具。系统不仅能够清晰地展示当月日期&#xff0c;还支持事件的添加、编辑和删除操作&#…

车载诊断架构SOVD --- 车辆发现与建连

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 钝感力的“钝”,不是木讷、迟钝,而是直面困境的韧劲和耐力,是面对外界噪音的通透淡然。 生活中有两种人,一种人格外在意别人的眼光;另一种人无论…

Notepad++找回自动暂存的文件

场景&#xff1a; 当你没有保存就退出Notepad&#xff0c;下次进来Notepad会自动把你上次编辑的内容显示出来&#xff0c;以便你继续编辑。除非你手动关掉当前页面&#xff0c;这样Notepad就会删除掉自动保存的内容。 问题&#xff1a; Notepad会将自动保存的文件地址,打开Note…

DL00924-基于深度学习YOLOv11的工程车辆目标检测含数据集

文末有代码完整出处 &#x1f697; 基于深度学习YOLOv11的工程车辆目标检测——引领智能识别新潮流&#xff01; &#x1f680; 随着人工智能技术的飞速发展&#xff0c; 目标检测 已经在各个领域取得了显著突破&#xff0c;尤其是在 工程车辆识别 这一关键技术上。今天&#…

Axure RP11安装、激活、汉化

一:注册码 Axure RP11.0.0.4122在2025-5-29日亲测有效: 49bb9513c40444b9bcc3ce49a7a022f9

自编码器Auto-encoder(李宏毅)

目录 编码器的概念&#xff1a; 为什么需要编码器&#xff1f; 编码器什么原理&#xff1f; 去噪自编码器: 自编码器的应用&#xff1a; 特征解耦 离散隐表征 编码器的概念&#xff1a; 重构&#xff1a;输入一张图片&#xff0c;通过编码器转化成向量&#xff0c;要求再…

数据结构之堆(topk问题、堆排序)

一、堆的初步认识 堆虽然是用数组存储数据的数据结构&#xff0c;但是它的底层却是另一种表现形式。 堆分为大堆和小堆&#xff0c;大堆是所有父亲大于孩子&#xff0c;小堆是所有孩子大于父亲。 通过分析我们能得出父子关系的计算公式&#xff0c;parent(child-1)/2&#xff…

SpringBoot使用ffmpeg实现视频压缩

ffmpeg简介 FFmpeg 是一个开源的跨平台多媒体处理工具集&#xff0c;用于录制、转换、编辑和流式传输音频和视频。它功能强大&#xff0c;支持几乎所有常见的音视频格式&#xff0c;是多媒体处理领域的核心工具之一。 官方文档&#xff1a;https://ffmpeg.org/documentation.h…

2025-05-31 Python深度学习9——网络模型的加载与保存

文章目录 1 使用现有网络2 修改网络结构2.1 添加新层2.2 替换现有层 3 保存网络模型3.1 完整保存3.2 参数保存&#xff08;推荐&#xff09; 4 加载网络模型4.1 加载完整模型文件4.2 加载参数文件 5 Checkpoint5.1 保存 Checkpoint5.2 加载 Checkpoint 本文环境&#xff1a; Py…

长安链起链调用合约时docker ps没有容器的原因

在调用这个命令的时候&#xff0c;发现并没有出现官方预期的合约容器&#xff0c;这是因为我们在起链的时候没有选择用docker的虚拟环境&#xff0c;实际上这不影响后续的调用&#xff0c;如果想要达到官方的效果那么你只需要在起链的时候输入yes即可&#xff0c;如图三所示

Appium+python自动化(七)- 认识Appium- 上

简介 经过前边的各项准备工作&#xff0c;终于才把appium搞定。 一、appium自我介绍 appium是一款开源的自动化测试工具&#xff0c;可以支持iOS和安卓平台上的原生的&#xff0c;基于移动浏览器的&#xff0c;混合的应用&#xff08;APP&#xff09;。 1、 使用appium进…

模块联邦:更快的微前端方式!

什么是模块联邦 在前端项目中&#xff0c;不同团队之间的业务模块可能有耦合&#xff0c;比如A团队的页面里有一个富文本模块&#xff08;组件&#xff09;&#xff0c;而B团队 的页面恰好也需要使用这个富文本模块。 传统模式下&#xff0c;B团队只能去抄A团队的代码&#x…

前端基础学习html+css+js

HTML 区块 div标签&#xff0c;块级标签 span包装小部分文本&#xff0c;行内元素 表单 CSS css选择器 css属性 特性blockinlineinline-block是否换行✅ 换行❌ 不换行❌ 不换行可设置宽高✅ 支持❌ 不支持✅ 支持常见元素div容器 p段落 h标题span文本容器 a超链接img图片…

手机打电话时将对方DTMF数字转为RFC2833发给局域网SIP坐席

手机打电话时将对方DTMF数字转为RFC2833发给局域网SIP坐席 --局域网SIP坐席呼叫 上一篇&#xff1a;手机打电话时由对方DTMF响应切换多级IVR语音菜单&#xff08;完结&#xff09; 下一篇&#xff1a;安卓App识别手机系统弹授权框包含某段文字-并自动点击确定按钮 一、前言 …

SAP Business One:无锡哲讯科技助力中小企业数字化转型的智慧之选

数字化转型&#xff0c;中小企业的必经之路 在当今竞争激烈的商业环境中&#xff0c;数字化转型已不再是大型企业的专利&#xff0c;越来越多的中小企业开始寻求高效、灵活的管理系统来优化业务流程、提升运营效率。作为全球领先的企业管理软件&#xff0c;SAP Business One…

小型语言模型:为何“小”才是“大”?

当说到人工智能&#xff08;AI&#xff09;的时候&#xff0c;大家通常会想到那些拥有数十亿参数的超大型语言模型&#xff0c;它们能做出一些令人惊叹的事情。 厉害不厉害&#xff1f;绝对厉害&#xff01; 但对于大多数企业和开发者来说&#xff0c;实用吗&#xff1f;可能…

秋招Day12 - 计算机网络 - 网络综合

从浏览器地址栏输入URL到显示网页的过程了解吗&#xff1f; 从在浏览器地址栏输入 URL 到显示网页的完整过程&#xff0c;并不是一个单一的数据包从头到尾、一次性地完成七层封装再七层解析的过程。 而是涉及到多次、针对不同目的、与不同服务器进行的、独立的网络通信交互&a…