【设计模式】享元模式(轻量级模式) 单纯享元模式和复合享元模式

享元模式(Flyweight Pattern)详解


一、享元模式简介

享元模式(Flyweight Pattern) 是一种 结构型设计模式(对象结构型模式),它通过共享技术实现相同或相似对象的重用,以减少内存占用和提高性能。具体来说,就是将一些细粒度的对象集中起来,使得这些对象可以互相共享。

又称为轻量级模式,要求能够被共享的对象必须是细粒度对象

运用共享技术有效地支持大量细粒度对象的复用。

简单来说:

“如果你有一堆相似的对象,为什么不让他们共享数据,从而节省内存呢?”
字符享元对象示意图
在这里插入图片描述

享元模式:通过共享技术实现相同或相似对象的重用
享元池(Flyweight Pool):存储共享实例对象的地方

原理

  1. 将具有相同内部状态的对象存储在享元池中,享元池中的对象是可以实现共享的
  2. 需要的时候将对象从享元池中取出,即可实现对象的复用
  3. 通过向取出的对象注入不同的外部状态,可以得到一系列相似的对象,而这些对象在内存中实际上只存储一份

享元模式包含以下4个角色
Flyweight(抽象享元类)
ConcreteFlyweight(具体享元类)
UnsharedConcreteFlyweight(非共享具体享元类)
FlyweightFactory(享元工厂类)

在这里插入图片描述


二、解决的问题类型

享元模式主要用于解决以下问题:

  • 大量小对象导致内存消耗过大:当你的应用程序需要创建大量相似的小对象时,直接创建这些对象会导致内存使用过高。
  • 提升性能:通过共享相同的对象实例来减少对象创建的数量,进而减少垃圾回收的压力,提高系统性能。
  • 对象的大部分状态都可以外部化,可以将这些外部状态传入对象中
  • 在使用享元模式时需要维护一个存储享元对象的享元池,而这需要耗费一定的系统资源,因此,在需要多次重复使用享元对象时才值得使用享元模式

三、使用场景

场景示例
字符串池Java 中的 String.intern() 方法
缓存管理如缓存数据库查询结果
文档处理文本编辑器中的字符样式管理
游戏开发游戏中大量的相同纹理、模型等资源

四、核心概念

1. 内部状态 vs 外部状态

  • 内部状态:存储在享元对象内部,可以在多个对象之间共享的部分,且不会随环境改变而改变的状态,通常是不变的数据。(例如:字符的内容)
  • 外部状态:不能被共享的部分,是随环境变化而变化的数据。享元对象的外部状态通常由客户端保存,并在享元对象被创建之后,需要使用的时候再传入到享元对象内部。一个外部状态与另一个外部状态之间是相互独立的(例如:字符的颜色和大小)

享元模式的核心思想就是分离对象的内部状态和外部状态,让内部状态可以共享,而外部状态则由客户端负责维护。


五、代码案例(Java)

我们以一个简单的文本编辑器为例,展示如何利用享元模式来优化字体样式的管理。

1. 定义享元接口

interface CharacterStyle {void display(char character);
}

2. 创建具体的享元类

class CharacterStyleImpl implements CharacterStyle {private final String fontName;private final int fontSize;public CharacterStyleImpl(String fontName, int fontSize) {this.fontName = fontName;this.fontSize = fontSize;}@Overridepublic void display(char character) {System.out.println(character + " in " + fontName + ", size: " + fontSize);}
}

3. 创建享元工厂

class CharacterStyleFactory {private static final Map<String, CharacterStyle> styles = new HashMap<>();public static CharacterStyle getStyle(String fontName, int fontSize) {String key = fontName + "-" + fontSize;if (!styles.containsKey(key)) {styles.put(key, new CharacterStyleImpl(fontName, fontSize));}return styles.get(key);}
}

4. 使用享元模式

public class Client {public static void main(String[] args) {// 获取样式CharacterStyle style1 = CharacterStyleFactory.getStyle("Arial", 12);CharacterStyle style2 = CharacterStyleFactory.getStyle("Arial", 12);// 显示字符style1.display('A');  // 输出: A in Arial, size: 12style2.display('B');  // 输出: B in Arial, size: 12// 验证是否为同一实例System.out.println(style1 == style_Statics);  // 输出: true}
}

典型代码(c++)

典型的抽象享元类代码

abstract class Flyweight
{public abstract void Operation(string extrinsicState);
}

典型的具体享元类代码

class ConcreteFlyweight : Flyweight
{//内部状态intrinsicState作为成员变量,同一个享元对象其内部状态是一致的private string intrinsicState;public ConcreteFlyweight(string intrinsicState) {this.intrinsicState = intrinsicState;}//外部状态extrinsicState在使用时由外部设置,不保存在享元对象中,即使是同一个对象,在每一次调用时可以传入不同的外部状态public override void Operation(string extrinsicState) {//实现业务方法}	
}

典型的非共享具体享元类代码

class UnsharedConcreteFlyweight : Flyweight
{public override void Operation(string extrinsicState){//实现业务方法}
}

典型的享元工厂类代码

using System.Collections;
class FlyweightFactory
{//定义一个Hashtable用于存储享元对象,实现享元池private Hashtable flyweights = new Hashtable();public Flyweight GetFlyweight(string key){//如果对象存在,则直接从享元池获取if (flyweights.ContainsKey(key)){return (Flyweight)flyweights[key];}//如果对象不存在,先创建一个新的对象添加到享元池中,然后返回else{Flyweight fw = new ConcreteFlyweight("state");flyweights.Add(key,fw);return fw;}}
}

其他案例

  1. 某软件公司要开发一个围棋软件,其界面效果如下图所示
    在这里插入图片描述
    该软件公司开发人员通过对围棋软件进行分析发现,在图中,围棋棋盘中包含大量的黑子和白子,它们的形状、大小都一模一样,只是出现的位置不同而已。如果将每一个棋子都作为一个独立的对象存储在内存中,将导致该围棋软件在运行时所需内存空间较大,如何降低运行代价、提高系统性能是需要解决的一个问题。为了解决该问题,现使用享元模式来设计该围棋软件的棋子对象。

在这里插入图片描述
如何让相同的黑子或者白子能够多次重复显示但位于一个棋盘的不同地方?

解决方案:将棋子的位置定义为棋子的一个外部状态,在需要时再进行设置

引入外部状态之后的围棋棋子结构图
在这里插入图片描述

//Coordinates.cs
namespace FlyweightSample
{class Coordinates{private int x;private int y;public Coordinates(int x, int y){this.x = x;this.y = y;}public int X{get { return x; }set { x = value; }}   public int Y{get { return y; }set { y = value; }}}
}
//IgoChessman.cs
using System;
namespace FlyweightSample
{abstract class IgoChessman{public abstract string GetColor();public void Display(Coordinates coord){Console.WriteLine("棋子颜色:{0},棋子位置:{1},{2}", this.GetColor(),coord.X,coord.Y);	}}
}
  1. 共享网络设备(无外部状态)
    很多网络设备都是支持共享的,如交换机、集线器等,多台终端计算机可以连接同一台网络设备,并通过该网络设备进行数据转发,如图所示,现用享元模式模拟共享网络设备的设计原理
    在这里插入图片描述
  2. 共享网络设备(有外部状态)
    虽然网络设备可以共享,但是分配给每一个终端计算机的端口(Port)是不同的,因此多台计算机虽然可以共享同一个网络设备,但必须使用不同的端口。我们可以将端口从网络设备中抽取出来作为外部状态,需要时再进行设置
    在这里插入图片描述

六、优缺点分析

优点描述
节省内存减少了大量相似对象的创建,降低了内存使用
提高性能减少了对象创建的时间开销
支持大规模并发在多线程环境中也能很好地工作
其他外部状态相对独立,而且不会影响其内部状态,从而使得享元对象可以在不同的环境中被共享
缺点描述
复杂性增加分离内部状态和外部状态增加了设计复杂度
可能影响可读性对于不熟悉该模式的人来说,理解代码逻辑可能会变得困难
不适合频繁变化的状态如果大部分状态都是外部状态,则享元模式的优势无法体现。为了使对象可以共享,享元模式需要将享元对象的部分状态外部化,而读取外部状态将使得运行时间变长

七、与其他模式对比(补充)

模式名称目标
单例模式确保某个类只有一个实例,并提供全局访问点
原型模式通过克隆已有对象来创建新对象,强调复用
享元模式通过共享技术减少相似对象的创建,节约内存

八、最终小结

享元模式是一种非常有效的设计模式,特别适合那些:

  • 存在大量相似对象的应用场景;
  • 希望减少内存使用并提高性能的情况;
  • 可以明确区分内部状态和外部状态的对象。

掌握享元模式可以帮助你在处理大量相似对象时更加高效地管理内存资源,特别是在游戏开发、文档处理等领域。


📌 一句话总结:

享元模式就像是一个“共享池”,把那些可以共享的部分放在一起,避免重复创建,从而达到节省内存的目的。


推荐使用方式:

  • 当你发现你的应用中有许多相似但独立的对象时,考虑使用享元模式。
  • 注意合理划分内部状态和外部状态,确保设计简洁且易于维护。

九、扩展

单纯享元模式和复合享元模式

单纯享元模式
所有的具体享元类都是可以共享的,不存在非共享具体享元类

在这里插入图片描述
复合享元模式
将一些单纯享元对象使用组合模式加以组合
如果希望为多个内部状态不同的享元对象设置相同的外部状态,可以考虑使用复合享元模式

在这里插入图片描述
部分内容由AI大模型生成,注意识别!

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

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

相关文章

驱动开发_2.字符设备驱动

目录1. 什么是字符设备2. 设备号2.1 设备号概念2.2 通过设备号dev分别获取主、次设备号的宏函数2.3 主设备号的申请静态申请动态分配2.4 注销设备号3. 字符设备3.1 注册字符设备3.2 注销字符设备3.3 应用程序和驱动程序的关系3.4 file_opertaions结构体3.5 class_create3.6 创建…

直播推流技术底层逻辑详解与私有化实现方案-以rmtp rtc hls为例-优雅草卓伊凡

直播推流技术底层逻辑详解与私有化实现方案-以rmtp rtc hls为例-优雅草卓伊凡由于我们的甲方客户要开始为我们项目产品上加入私有化的直播&#xff0c;这块不得不又捡起来曾经我们做直播推流的事情了&#xff0c;其实私有化直播一直并不是一件容易的事情&#xff0c;现在大部分…

一文读懂现代卷积神经网络—深度卷积神经网络(AlexNet)

目录 深度卷积神经网络&#xff08;AlexNet&#xff09;是什么&#xff1f; 一、AlexNet 的核心创新 1. 深度架构 2. ReLU 激活函数 3. 数据增强 4. Dropout 正则化 5. GPU 并行计算 6. 局部响应归一化&#xff08;LRN&#xff09; 二、AlexNet 的网络结构 三、AlexN…

JVM 垃圾收集算法全面解析

1. 引言1.1 为什么需要垃圾收集&#xff1f;在Java应用中&#xff0c;垃圾收集&#xff08;Garbage Collection&#xff0c;GC&#xff09;是一个至关重要的机制&#xff0c;它使得开发者不需要手动管理内存。与传统的语言&#xff08;如C或C&#xff09;不同&#xff0c;Java通…

Vmware中安装的CentOS7如何扩展硬盘大小

起初创建虚拟机时&#xff0c;大小设置不合理&#xff0c;导致我在尝试开源项目时空间不足重新扩展硬盘&#xff0c;不仅需要在虚拟机设置中配置&#xff0c;还需要在系统内重新进行分区一、虚拟机设置打开虚拟机设置→硬盘→扩展&#xff0c;将大小设置为自己期望的大小&#…

Python+MongoDB高效开发组合

如大家所知&#xff0c;Python与MongoDB的结合是一种高效的开发组合&#xff0c;主要用于通过Python进行数据存储、查询及管理&#xff0c;利用MongoDB的文档型数据库特性实现灵活的数据处理。下面让 Python 连接上 MongoDB&#xff1a;安装 PyMongo&#xff1a;pip3 install p…

【论文阅读】Masked Autoencoders Are Effective Tokenizers for Diffusion Models

introduce什么样的 latent 空间更适合用于扩散模型&#xff1f;作者发现&#xff1a;相比传统的 VAE&#xff0c;结构良好、判别性强的 latent 空间才是 diffusion 成功的关键。研究动机&#xff1a;什么才是“好的 latent 表征”&#xff1f;背景&#xff1a;Diffusion Models…

每日一SQL 【游戏玩法分析 IV】

文章目录问题案例执行顺序使用分组解决问题 案例 执行顺序 SQL 语句的执行顺序&#xff08;核心步骤&#xff09; 同一层级的select查询内部, 别名在整个 SELECT 计算完成前不生效 使用分组解决 select distinct s.product_id, Product.product_name from Sales sleft join …

内部文件审计:企业文件服务器审计对网络安全提升有哪些帮助?

企业文件服务器审计工作不仅对提升企业网络信息安全起到重要作用&#xff0c;还能对企业内部网络文件信息是否合规进行判断。因此企业文件服务器审计一直被高度重视。 一、文件服务器为何成为攻击焦点&#xff1f; 企业文件服务器通常集中存储财务报表、人事档案、研发资料、客…

FusionOne HCI 23 超融合实施手册(超聚变超融合)

产品介绍 FusionOne HCI作为实现企业信息一体化的IT基础设施平台&#xff0c;以“软硬件垂直深度集成和调优”、“快速部署”、“统一管理”的理念&#xff0c;提供应用融合部署&#xff0c;提升核心业务运作效率&#xff0c;降低整体采购成本。 FusionOne HCI代表了IT产品的…

AI算姻缘测算小工具流量主微信小程序开源

功能特点 响应式设计&#xff1a;完美适配各种移动设备屏幕尺寸 精美UI界面&#xff1a; 柔和的粉红色渐变背景 圆角卡片设计 精心设计的字体和间距 爱心图标点缀 动态效果&#xff1a; 点击按钮时的动画反馈 测算结果的平滑过渡动画 爱心漂浮动画 进度条动态填充 AI测算功能&a…

Vue获取上传Excel文件内容并展示在表格中

一、安装依赖 npm install xlsx 二、引用依赖 import XLSX from xlsx 三、代码实现 1、注意&#xff1a;函数 analysis 中reader.readAsBinaryString(file)&#xff0c;file的数据格式如图所示 2、示例代码 <!-- 项目使用的前端框架为非流行框架&#xff0c;主要关注…

pipelineJob和pipeline的关系

pipelineJob与pipeline在Jenkins体系中构成配置层与执行层的协同关系,具体关联如下: 一、核心功能定位 概念作用实现层级pipelineJob定义Job的元数据(如SCM配置、日志策略)配置层pipeline描述实际构建流程(如阶段划分、并行任务)执行层scriptPath桥梁作用:将配置层定义…

第二十篇 Word文档自动化:Python批量生成、模板填充与内容修改,告别繁琐排版!

python实现word 自动化重复性文档制作&#xff0c;手动填充模板&#xff0c;效率低下还易错1.python-docx入门&#xff1a;Word文档的“瑞士军刀”&#xff01;1.1 安装与基础概念&#xff1a;文档、段落、运行、表格1.2 打开/创建Word文档&#xff1a;Python与Word的初次接触1…

【C# in .NET】7. 探秘结构体:值类型的典型代表

探秘结构体&#xff1a;值类型的典型代表 在 C# 的类型系统中&#xff0c;结构体&#xff08;Struct&#xff09;作为值类型的典型代表&#xff0c;一直扮演着既基础又微妙的角色。许多开发者在日常编码中虽频繁使用结构体&#xff08;如int、DateTime等&#xff09;&#xff0…

深入探讨Hadoop YARN Federation:架构设计与实践应用

Hadoop YARN Federation简介基本概念与设计初衷Hadoop YARN Federation作为Apache Hadoop 3.x版本的核心特性之一&#xff0c;其本质是通过多集群联合管理机制突破单点资源管理器的性能瓶颈。传统YARN架构中&#xff0c;单个ResourceManager&#xff08;RM&#xff09;需要管理…

STM32固件升级设计——SD卡升级固件

目录 概述 一、功能描述 1、BootLoader部分&#xff1a; 2、APP部分&#xff1a; 二、BootLoader程序制作 1、分区定义 2、 主函数 3、SD卡升级文件检测和更新 4、程序跳转 三、APP程序制作 四、工程配置&#xff08;默认KEIL5&#xff09; 五、运行测试 结束语…

基于Python的图像文字识别系统

主要语言&#xff1a;Python数据库&#xff1a;SQLiteUI界面&#xff1a;PYQT5文字识别模型&#xff1a;Tesseract OCR&#xff08;本地搭建&#xff09;主要功能&#xff1a;登录注册&#xff1a;登录注册功能。图片管理&#xff1a;单张/多张上传、图片列表、预览、删除、切换…

028_分布式部署架构

028_分布式部署架构 概述 本文档介绍如何设计和实现Claude应用的分布式部署架构&#xff0c;包括负载均衡、缓存策略、服务发现、容错机制等。 微服务架构设计 1. 服务拆分策略 from abc import ABC, abstractmethod from typing import Dict, Any, Optional import asyncio im…

duckdb和pyarrow读写arrow格式的方法

arrow格式被多种分析型数据引擎广泛采用&#xff0c;如datafusion、polars。duckdb有一个arrow插件&#xff0c;原来是core插件&#xff0c;1.3版后被废弃&#xff0c;改为社区级插件&#xff0c;名字改为nanoarrow, 别名还叫arrow。 安装 D install arrow from community; D…