0

0

C++享元模式节省大量对象内存使用

P粉602998670

P粉602998670

发布时间:2025-09-02 10:14:02

|

163人浏览过

|

来源于php中文网

原创

c++享元模式节省大量对象内存使用

C++中的享元模式,说白了,就是一种聪明地节省内存的策略,尤其是在你的程序需要创建大量相似对象时。它通过识别并共享那些对象之间不变的、内在的数据(我们称之为“享元”),避免了为每个对象都复制一份相同的数据,从而显著减少了内存占用。那些会变化的数据,也就是“外在状态”,则被分离出来,由客户端或者上下文来维护。

享元模式的核心思想是“分离”与“共享”。想象一下一个文本编辑器,里面有成千上万个字符。如果每个字符对象都包含自己的字体、大小、颜色信息,那内存消耗会非常巨大。但实际上,很多字符可能共享相同的字体、大小和颜色。享元模式就是抓住这一点:把字符的字体、大小、颜色(内在状态,可共享)抽离出来,只创建一份,然后让所有使用相同属性的字符都指向这份共享数据。而每个字符独有的位置信息(外在状态,不可共享)则由外部传入或存储。

实现上,通常会有一个“享元工厂”(FlyweightFactory),它负责管理和创建享元对象。当你需要一个特定内在状态的享元时,工厂会先检查是否已经存在一个具有相同内在状态的享元。如果存在,就直接返回已有的;如果不存在,就创建一个新的并缓存起来。这样就确保了每种内在状态只有一个实例。

#include 
#include 
#include 
#include  // 使用智能指针更好地管理内存

// 享元接口
class CharacterFlyweight {
public:
    virtual void display(int x, int y) const = 0; // x, y 是外在状态
    virtual ~CharacterFlyweight() = default;
};

// 具体享元:包含内在状态 (字符本身, 字体, 大小, 颜色)
class ConcreteCharacterFlyweight : public CharacterFlyweight {
private:
    char character;
    std::string font;
    int size;
    std::string color;

public:
    ConcreteCharacterFlyweight(char c, const std::string& f, int s, const std::string& col)
        : character(c), font(f), size(s), color(col) {
        std::cout << "创建享元: " << character << " [" << font << ", " << size << ", " << color << "]\n";
    }

    void display(int x, int y) const override {
        std::cout << "显示字符 '" << character << "' 在 (" << x << "," << y << "),使用 ["
                  << font << ", " << size << ", " << color << "]\n";
    }
};

// 享元工厂:管理享元对象
class CharacterFlyweightFactory {
private:
    std::map> flyweights; // key可以是内在状态的组合

    // 辅助函数,用于生成享元对象的唯一键
    std::string getKey(char c, const std::string& f, int s, const std::string& col) const {
        return std::string(1, c) + "_" + f + "_" + std::to_string(s) + "_" + col;
    }

public:
    std::shared_ptr getFlyweight(char c, const std::string& f, int s, const std::string& col) {
        std::string key = getKey(c, f, s, col);
        if (flyweights.find(key) == flyweights.end()) {
            flyweights[key] = std::make_shared(c, f, s, col);
        }
        return flyweights[key];
    }

    // 析构函数不再需要手动删除,因为使用了shared_ptr
    // ~CharacterFlyweightFactory() { ... }
};

/*
// 客户端使用示例
int main() {
    CharacterFlyweightFactory factory;

    // 假设我们有文本 "Hello World"
    // 'H'
    std::shared_ptr h1 = factory.getFlyweight('H', "Arial", 12, "Black");
    h1->display(10, 10);

    // 'e'
    std::shared_ptr e1 = factory.getFlyweight('e', "Arial", 12, "Black");
    e1->display(20, 10);

    // 'l'
    std::shared_ptr l1 = factory.getFlyweight('l', "Arial", 12, "Black");
    l1->display(30, 10);

    // 另一个 'l',会重用上面的享元,因为内在状态相同
    std::shared_ptr l2 = factory.getFlyweight('l', "Arial", 12, "Black");
    l2->display(40, 10); // 注意:l1和l2指向同一个享元对象

    // 'o'
    std::shared_ptr o1 = factory.getFlyweight('o', "Arial", 12, "Black");
    o1->display(50, 10);

    // 改变字体和颜色,会创建新的享元
    std::shared_ptr h2 = factory.getFlyweight('H', "Times New Roman", 14, "Red");
    h2->display(10, 30);

    return 0;
}
*/

在上面的例子里,

ConcreteCharacterFlyweight
是享元,它的
character
font
size
color
是内在状态。
display
方法接收
x
y
作为外在状态,这些是每个字符实例独有的。
CharacterFlyweightFactory
负责确保每种内在状态组合只对应一个
ConcreteCharacterFlyweight
实例。这样,即使你的文本有上万个'l',如果它们都用"Arial"、12号字、黑色,内存中也只会有一份'l'的享元对象。这里我特意用了
std::shared_ptr
来管理享元对象的生命周期,让工厂的析构变得更简单,也更符合现代C++的实践。

立即学习C++免费学习笔记(深入)”;

C++享元模式如何有效降低大规模对象内存开销?

享元模式之所以能显著降低内存开销,主要在于它巧妙地利用了“重复”这个特性。在许多应用中,我们常常会创建大量本质上非常相似的对象。这些对象可能在某些属性上完全一致,只是在另一些属性上有所不同。享元模式的魔力就在于它能够识别出那些“内在的、共享的”属性,并将它们从每个对象实例中剥离出来,统一存储一份。

比如说,在游戏开发中,场景里可能有成千上万棵树。如果每棵树对象都包含了树的模型数据、纹理、材质等信息,那内存肯定会爆炸。但实际上,这些树可能都属于同一种“橡树”或“松树”,它们共享相同的模型和纹理。享元模式就是把这些共享的几何数据、纹理数据作为享元对象,每个具体的“树实例”只存储自己的位置、旋转、缩放等“外在状态”,然后引用那个共享的“橡树模型享元”。这样一来,无论场景里有多少棵橡树,内存中只需要一份橡树的模型数据。

B2S商城系统
B2S商城系统

B2S商城系统B2S商城系统是由佳弗网络工作室凭借专业的技术、丰富的电子商务经验在第一时刻为最流行的分享式购物(或体验式购物)推出的开源程序。开发采用PHP+MYSQL数据库,独立编译模板、代码简洁、自由修改、安全高效、数据缓存等技术的应用,使其能在大浏览量的环境下快速稳定运行,切实节约网站成本,提升形象。注意:如果安装后页面打开出现找不到数据库等错误,请删除admin下的runtime文件夹和a

下载

这种方法避免了大量重复数据的存储。当系统需要创建新对象时,它不再是完整地复制一份所有数据,而是先去享元工厂看看有没有现成的、符合内在状态的享元。有就直接用,没有才创建。这就像图书馆里的书,虽然有很多人借阅,但图书馆里每本书通常只有一两本,而不是每个读者都发一本全新的。这样就大大减少了物理存储空间的需求。当然,这并不是没有代价的,我们后面会提到。但对于内存敏感型应用,这种节省是相当可观的。

享元模式在设计和实现时有哪些关键考量点?

设计和实现享元模式,需要一些深思熟虑。首先,也是最关键的,就是如何清晰地划分内在状态和外在状态。内在状态是那些可以被多个对象共享的、不随上下文改变的数据,它通常是构造享元对象的关键。外在状态则是那些随上下文变化、不能共享的数据,它通常通过方法参数传递给享元对象。这个划分如果错了,享元模式就可能适得其反,或者根本无法工作。我个人经验是,这往往是享元模式最考验设计功力的地方。

其次,享元工厂的实现至关重要。工厂需要一个高效的查找机制来判断某个内在状态的享元是否已经存在。通常会使用

std::map
std::unordered_map
,以内在状态的某个唯一标识(比如字符串哈希,或者一个结构体作为key)作为键,存储享元对象的智能指针。这里要考虑到键的生成成本和查找效率。如果键的生成或查找过于复杂和耗时,可能会抵消掉内存节省带来的性能优势。

再者,享元对象的生命周期管理也是一个需要注意的地方。享元对象通常由工厂创建并拥有,直到程序结束或不再需要时才销毁。在现代C++中,使用

std::shared_ptr
是一个非常好的选择,它能自动管理享元对象的生命周期,避免了手动内存管理可能带来的泄漏问题。工厂只需持有
shared_ptr
,当所有引用都消失时,对象就会被正确销毁。

最后,线程安全。如果享元工厂可能被多个线程同时访问,那么对享元缓存(

std::map
)的访问就需要进行同步,比如使用互斥锁(
std::mutex
)。否则,可能会导致竞态条件

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1497

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

623

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

592

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

586

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

170

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

82

2025.08.07

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

1

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

相关下载

更多

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号