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

设计模式[11]——享元模式一分钟彻底说清楚

设计模式[11]——享元模式一分钟彻底说清楚

一句话定义

通过共享大量细粒度对象的内在状态(不变部分),大幅减少内存占用,让成千上万个相似对象只占用少量内存。

最狠的比喻(软件人专属)

游戏里渲染一片森林:

  • 有10万棵树
  • 树只有5种模型(松树、橡树、桦树、棕榈、樱花树)
  • 每棵树的位置、大小、旋转角度不同(外在状态)
  • 但模型网格、纹理、材质完全相同(内在状态)

不共享:10万份完整模型 → 内存爆炸
享元模式:只存5份模型,所有树共享 → 内存节省99%!

为什么需要它?(坏味道瞬间爆炸)

不用享元,你会这样写:

structTree{Mesh mesh;// 每个树都完整复制一份网格(几MB)Texture texture;Vector3 position;floatscale;};// 10万棵树 = 几GB内存,寄!
和之前模式彻底分清(10秒表)
项目装饰器(Decorator)组合(Composite)外观(Facade)享元(Flyweight)
核心意图动态叠加行为部分-整体统一接口简化复杂子系统共享内在状态节省内存
关键机制包装链树形递归统一入口工厂 + 共享池
对象数量少量中等(树节点)单个外观大量细粒度对象
典型场景流加密/日志UI树/场景图视频转码/编译器游戏渲染、文字处理、粒子系统
口号“层层叠加”“套娃统一”“一键搞定”“千树一面,共享内在”
真实软件例子:游戏场景树渲染(Unreal/Unity风格)
#include<iostream>#include<memory>#include<unordered_map>#include<vector>#include<string>usingnamespacestd;// 1. 享元接口(内在状态只读)classTreeModel{public:virtual~TreeModel()=default;virtualvoidrender(floatx,floaty,floatscale)const=0;virtualstringtype()const=0;};// 2. 具体享元(内在状态:网格、纹理等昂贵资源)classPineTree:publicTreeModel{public:voidrender(floatx,floaty,floatscale)constoverride{cout<<"[共享模型] 松树 @ ("<<x<<","<<y<<") 缩放:"<<scale<<endl;}stringtype()constoverride{return"Pine";}};classOakTree:publicTreeModel{public:voidrender(floatx,floaty,floatscale)constoverride{cout<<"[共享模型] 橡树 @ ("<<x<<","<<y<<") 缩放:"<<scale<<endl;}stringtype()constoverride{return"Oak";}};// 3. 享元工厂(核心:缓存共享对象)classTreeModelFactory{unordered_map<string,unique_ptr<TreeModel>>models;public:TreeModel*getModel(conststring&type){if(!models.count(type)){cout<<"[工厂] 创建新共享模型: "<<type<<endl;if(type=="Pine")models[type]=make_unique<PineTree>();elseif(type=="Oak")models[type]=make_unique<OakTree>();// 真实项目:这里加载网格、纹理等大资源,只加载一次!}returnmodels[type].get();// 返回共享指针}};// 4. 外在状态(每个树实例独有,轻量)structTreeInstance{floatx,y;floatscale;TreeModel*model;// 指向共享的享元voidrender()const{model->render(x,y,scale);}};
客户端:10万棵树,内存只占5个模型
intmain(){TreeModelFactory factory;vector<TreeInstance>forest;// 生成10万棵树,只创建2种共享模型for(inti=0;i<100000;++i){string type=(i%2==0)?"Pine":"Oak";forest.push_back({float(i%1000),float(i/1000),0.8f+(i%3)*0.2f,factory.getModel(type)});}cout<<"\n=== 开始渲染森林 ===\n";for(inti=0;i<10;++i){// 只渲染前10棵演示forest[i].render();}cout<<"... 剩余99990棵树同样共享模型,内存爆炸?不存在的!\n";}

输出:

[工厂] 创建新共享模型: Pine [工厂] 创建新共享模型: Oak === 开始渲染森林 === [共享模型] 松树 @ (0,0) 缩放:0.8 [共享模型] 橡树 @ (1,0) 缩放:1 [共享模型] 松树 @ (2,0) 缩放:1.2 ...
C++ 真实项目里无处不在
  • 游戏引擎:Unreal的Foliage系统、Unity的Instancing渲染(共享Mesh和Material)
  • 文字渲染:每个字符(A~Z)只存一份Glyph(字体轮廓),成千上万文字实例共享
  • 粒子系统:10万粒子共享几种粒子纹理和行为
  • UI图标:整个App共享一套图标纹理图集(Texture Atlas)
  • Qt/OpenGL:共享VAO/VBO、Shader程序
经典坑 & 正确姿势
  • 内在状态必须不可变(共享对象不能被单个实例修改)
  • 外在状态由客户端持有(位置、颜色、缩放等)
  • 享元工厂通常是单例或静态
终极口诀(游戏开发者专属)

“千树万树同一模,内在共享外在独;
内存爆炸不存在,享元工厂真牛逼!”

刻在DNA里的一句话

当你面对“大量相似细粒度对象”(游戏实体、字符、粒子、图块),且内在状态远大于外在状态时,
立刻上享元模式——用工厂缓存共享对象,内存从GB降到MB!

现在,享元模式彻底说透了!
结构型模式还剩最后一篇:代理模式(Proxy)。

http://www.cnnetsun.cn/news/106834.html

相关文章:

  • 多人姿态估计终极指南:从零开始构建实时人体分析系统
  • 【ACWing】150. 括号画家
  • 如何快速掌握Vim插件管理:VAM的完整使用指南
  • 文献分区及影响因子批量查询
  • APKMirror安卓应用下载平台深度解析:从源码到实践
  • 终极FreeMarker模板调试工具:3分钟解决模板语法问题
  • QQScreenShot独立版技术解析:基于模块化架构的屏幕捕捉解决方案
  • 快速掌握SCPI Parser终极指南:构建专业仪器控制系统的完整解决方案
  • 自定义算子的“诞生记”:基于CANN Kernel自调工程的完整CI/CD流水线
  • 高效、稳定、可定制——EmotiVoice开源TTS优势全解析
  • 大模型应用开发(十八)_向量检索
  • NVIDIA显卡设置终极指南:从问题诊断到性能优化的完整解决方案
  • 聚星成链,蓝卓牵头成立“工厂操作系统生态联盟”共建产业新生态
  • 每天一道面试题之架构篇|可靠订单状态机与事务消息架构设计
  • 10分钟掌握开源美颜SDK核心技术:从算法原理到商业应用实战
  • EmotiVoice支持哪些语言?多语种语音合成能力测试报告
  • AI语音合成进入情感时代:EmotiVoice带来全新听觉体验
  • EmotiVoice支持WebAssembly吗?浏览器端运行可能性分析
  • StaMPS雷达数据处理:从零搭建专业位移监测系统
  • yt-dlp-gui终极指南:轻松掌握Windows视频下载利器
  • EmotiVoice是否支持语音情感随机扰动?增强自然感功能
  • QRemeshify终极指南:快速创建高质量四边形网格的完整教程
  • 如何免费获得高质量语音合成能力?EmotiVoice给你答案
  • Hive SQL中COALESCE 函数和NVL()函数、IFNULL函数区别
  • 四边形网格生成实战指南:掌握QuadriFlow高效工作流
  • 如何快速解决AMD GPU识别问题:终极故障排查指南
  • OpenProject企业版深度解析:从开源到商业化的全面升级
  • Next.js认证系统实战:基于Clerk的完整解决方案
  • DeepBench如何帮助你在5分钟内完成深度学习硬件性能精准评估?
  • PCB文件处理终极指南:用Python轻松解析Gerber和Excellon文件