首页 > 后端开发 > C++ > 正文

c++如何实现一个享元模式_c++结构型设计模式Flyweight【性能优化】

php中文网
发布: 2025-12-13 09:39:50
原创
810人浏览过
享元模式通过共享内在状态、分离外在状态来减少内存占用;核心是用工厂缓存并复用相同内在状态的对象,客户端传入外在状态调用行为。

c++如何实现一个享元模式_c++结构型设计模式flyweight【性能优化】

享元模式(Flyweight)在 C++ 中主要用于减少内存占用,通过共享大量细粒度对象来提升性能,特别适用于对象数量庞大、状态可分“内在”与“外在”的场景(比如文字编辑器中的字符、游戏中的子弹、GUI 中的图标等)。

核心思想:分离内在状态与外在状态

内在状态(Intrinsic State)是共享的、不可变的,属于享元对象自身,比如字符的字体、大小、颜色(若统一);
外在状态(Extrinsic State)是不共享的、变化的,由客户端维护并传入,比如字符在文档中的位置、行号、是否选中等。

关键不是“把对象变少”,而是“避免重复创建相同内在状态的对象”。

手动实现享元工厂(推荐初学理解)

std::mapstd::unordered_map 缓存已创建的享元实例,按内在状态作 key 查找复用:

  • 定义享元基类(接口或抽象类),声明操作函数,参数接收外在状态
  • 派生具体享元类,只保存内在状态(const 成员),实现逻辑
  • 实现享元工厂类,负责创建/获取享元,内部用 map 管理生命周期(注意线程安全需加锁)
  • 客户端不再 new 具体类,而是调用工厂 getFlyweight(intrinsicKey),再传 extrinsic 参数调用其方法

示例关键片段:

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

星辰Agent
星辰Agent

科大讯飞推出的智能体Agent开发平台,助力开发者快速搭建生产级智能体

星辰Agent 378
查看详情 星辰Agent
class CharacterFlyweight {
public:
    virtual void render(int x, int y) const = 0; // 外在状态:坐标
    virtual ~CharacterFlyweight() = default;
};

class ConcreteChar : public CharacterFlyweight {
    const char ch_;
    const std::string font_;
    const int size_;
public:
    ConcreteChar(char c, std::string f, int s) 
        : ch_(c), font_(std::move(f)), size_(s) {}
    
    void render(int x, int y) const override {
        std::cout << "Draw '" << ch_ << "' at (" << x << "," << y << ") "
                  << "with " << font_ << " " << size_ << "pt\n";
    }
};

class FlyweightFactory {
    std::unordered_map<std::string, std::unique_ptr<CharacterFlyweight>> pool;
public:
    CharacterFlyweight& getFlyweight(const std::string& key) {
        auto it = pool.find(key);
        if (it != pool.end()) return *it->second;
        
        // 解析 key(如 "A|Arial|12")→ 构造对象
        auto pos1 = key.find('|'), pos2 = key.find('|', pos1+1);
        char c = key[0];
        std::string font = key.substr(pos1+1, pos2-pos1-1);
        int size = std::stoi(key.substr(pos2+1));
        
        pool[key] = std::make_unique<ConcreteChar>(c, font, size);
        return *pool[key];
    }
};
登录后复制

使用智能指针 + 工厂封装(更现代、安全)

避免裸指针和手动管理,用 std::shared_ptr 管理享元生命周期,工厂返回 const 引用或 shared_ptr:

  • 享元类建议设计为轻量、无状态修改(所有成员 const 或 immutable)
  • 工厂内部用 static 局部变量 + unordered_map 实现单例式缓存,天然线程安全(C++11 起)
  • 客户端拿到的是共享指针,无需关心释放;也可返回 const CharacterFlyweight& 提高性能(但需确保工厂生命周期长于使用者)

什么时候不该用享元?

享元不是银弹,滥用反而降低可读性和性能:

  • 对象总数很少(
  • 内在状态组合爆炸(比如 10 种字体 × 5 种大小 × 20 种颜色 → 1000 种 key),缓存反而吃内存
  • 外在状态传递复杂,导致接口臃肿、调用频繁出错
  • 对象本身已很轻量(如仅含一个 int),共享意义不大

基本上就这些。享元本质是空间换时间 + 状态解耦,C++ 实现关键在于明确区分内外状态、合理设计 key、用好容器和智能指针。不复杂但容易忽略外在状态的归属和线程安全性。

以上就是c++++如何实现一个享元模式_c++结构型设计模式Flyweight【性能优化】的详细内容,更多请关注php中文网其它相关文章!

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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