设计模式[11]——享元模式一分钟彻底说清楚
2026/4/25 7:22:46 网站建设 项目流程

设计模式[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)。

需要专业的网站建设服务?

联系我们获取免费的网站建设咨询和方案报价,让我们帮助您实现业务目标

立即咨询