C++怎么实现一个享元模式(Flyweight)_C++设计模式与享元模式实现

享元模式通过共享内部状态减少内存开销,分离可变外部状态与不可变内部状态。示例中TreeType存储种类、颜色、纹理等内部状态,由TreeFactory管理复用;位置作为外部状态在draw时传入。Forest中种植多棵树,相同类型的树共享同一TreeType实例,避免重复创建,显著降低内存使用,适用于对象数量庞大且存在重复数据的场景。

c++怎么实现一个享元模式(flyweight)_c++设计模式与享元模式实现

享元模式(Flyweight Pattern)是一种结构型设计模式,主要用于减少创建大量相似对象时的内存开销。它的核心思想是:通过共享尽可能多的数据来支持大量细粒度的对象。在C++中,实现享元模式的关键在于分离**内部状态(Intrinsic State)** 和 **外部状态(Extrinsic State)**。

内部状态是可共享的、不会随环境改变的状态;外部状态是依赖上下文、不可共享的部分,通常由客户端传入。

享元模式的基本结构

一个典型的享元模式包含以下几个部分:

Flyweight(抽象享元类):定义接口,接受外部状态作为参数。ConcreteFlyweight(具体享元类):实现 Flyweight 接口,并存储内部状态。UnsharedConcreteFlyweight(非共享具体享元):不需要共享的对象,可选。FlyweightFactory(享元工厂):负责管理享元对象,确保共享对象能被正确复用。

C++ 实现示例

假设我们要绘制森林中的树,每棵树有类型(种类、颜色、纹理)和位置(X, Y)。其中“种类、颜色、纹理”是内部状态,可以共享;“位置”是外部状态,每次调用时传入。

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

// TreeType.h – 共享的内部状态class TreeType {private:std::string name;std::string color;std::string texture;

public:TreeType(const std::string& n, const std::string& c, const std::string& t): name(n), color(c), texture(t) {}

void draw(int x, int y) const {    std::cout << "Drawing " << name << " tree at (" << x << "," << y               << ") with color " << color << " and texture " << texture << "n";}

};

// TreeFactory.h – 享元工厂class TreeFactory {private:std::vector> treeTypes;

public:TreeType* getTreeType(const std::string& name, const std::string& color, const std::string& texture) {for (const auto& tt : treeTypes) {if (tt->getName() == name && tt->getColor() == color && tt->getTexture() == texture) {return tt.get();}}// 没找到就创建新的treeTypes.push_back(std::make_unique(name, color, texture));return treeTypes.back().get();}};

上面代码中我们省略了 TreeType 的 getter 方法,实际使用中需要添加 getName(), getColor(), getTexture() 等函数用于比较。

// Tree.h – 外部接口,使用享元class Tree {private:int x, y; // 外部状态TreeType* type; // 内部状态(共享)

public:Tree(int x, int y, TreeType* type) : x(x), y(y), type(type) {}

void draw() const {    type->draw(x, y);}

};

// Forest.h – 容器类class Forest {private:std::vector> trees;TreeFactory factory;

public:void plantTree(int x, int y, const std::string& name, const std::string& color, const std::string& texture) {TreeType* type = factory.getTreeType(name, color, texture);trees.push_back(std::make_unique(x, y, type));}

void draw() const {    for (const auto& tree : trees) {        tree->draw();    }}

};

使用示例

int main() { Forest forest;

forest.plantTree(1, 2, "Pine", "Green", "Needle");forest.plantTree(3, 4, "Pine", "Green", "Needle");  // 复用同一类型forest.plantTree(5, 6, "Oak", "Brown", "Broad");forest.draw();return 0;

}

输出结果:

Drawing Pine tree at (1,2) with color Green and texture NeedleDrawing Pine tree at (3,4) with color Green and texture NeedleDrawing Oak tree at (5,6) with color Brown and texture Broad

虽然创建了三棵树,但只生成了两个 TreeType 对象,“Pine” 类型被成功复用。

关键点总结

实现享元模式需要注意以下几点:

把不变的状态提取出来作为享元对象,通常是构造参数固定的资源。享元对象必须是不可变的或线程安全的,否则共享会出问题。工厂类应保证相同参数只创建一次享元对象,可用哈希表优化查找。外部状态不能保存在享元中,必须在方法调用时传入。

基本上就这些。享元模式适合对象数量巨大且存在大量重复数据的场景,比如文字编辑器中的字符格式、游戏中的粒子系统或地图元素。合理使用能显著降低内存占用

以上就是C++怎么实现一个享元模式(Flyweight)_C++设计模式与享元模式实现的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1485040.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 08:35:18
下一篇 2025年12月14日 09:25:13

相关推荐

  • c++怎么链接一个静态库或者动态库_c++库文件引用与链接方式详解

    静态库在编译时嵌入可执行文件,动态库运行时加载;使用-L指定库路径,-l链接库名,-I包含头文件路径,确保环境变量或系统配置正确以避免链接和运行时错误。 在C++开发中,使用静态库或动态库可以有效复用代码。链接库文件是编译过程中的关键步骤,理解如何正确引用和链接库对项目构建至关重要。 静态库与动态库…

    2025年12月19日
    000
  • C++如何使用std::atomic_C++原子操作与多线程安全实践

    std::atomic是C++11引入的模板类,用于保证对基本类型的读写操作具有原子性,避免多线程下的数据竞争。它通过提供load、store、fetch_add、exchange和compare_exchange_weak/strong等原子操作,实现无锁并发控制。相比互斥锁,std::atomi…

    2025年12月19日
    000
  • C++怎么使用map和unordered_map_C++关联容器的性能与选择

    map基于红黑树实现,有序且操作复杂度为O(log n);unordered_map基于哈希表,无序但平均O(1)操作,适合快速查找。选择依据:需排序用map,求速度且无需序用unordered_map。 在C++中,map 和 unordered_map 是两种常用的关联容器,用于存储键值对(ke…

    2025年12月19日
    000
  • c++中的命名返回值优化(NRVO)是什么_c++中命名返回值优化(NRVO)原理与应用示例

    NRVO是C++中通过直接在调用方内存构造具名返回对象来消除拷贝开销的优化技术,当函数单一返回同一名对象时,编译器可将其构造于预分配的目标位置,避免临时对象创建与复制,提升性能。 命名返回值优化(Named Return Value Optimization, 简称 NRVO)是 C++ 编译器提供…

    2025年12月19日
    000
  • C++中的异常安全保证是什么_C++异常处理与异常安全策略

    异常安全保证确保C++程序在抛出异常时仍保持有效状态,避免资源泄漏或数据损坏。它分为三个级别:基本保证、强保证和无抛出保证。基本保证指对象处于有效但不可预测的状态;强保证要求操作原子性,成功则完全生效,失败则回滚;无抛出保证则确保操作绝不抛出异常。为实现这些级别,应采用RAII管理资源,使用智能指针…

    2025年12月19日
    000
  • 在嵌入式系统中使用C++构建高可靠性应用

    在嵌入式系统中使用c++++构建高可靠性应用是现代工业、医疗、汽车和航空航天等领域的重要趋势。尽管传统上嵌入式开发多采用c语言,但c++在保持性能的同时提供了更强的抽象能力和代码组织结构,有助于提升系统的可维护性和可靠性。关键在于合理使用c++特性,规避潜在风险。 选择性使用C++特性以控制复杂性 …

    好文分享 2025年12月19日
    000
  • 嵌入式系统开发中实现模块化C++架构设计方法

    在嵌入式系统开发中,c++++ 的模块化架构设计能显著提升代码的可维护性、可重用性和可测试性。尽管嵌入式环境资源受限,合理使用 c++ 特性仍可在不牺牲性能的前提下实现良好的模块划分。关键在于结合面向对象设计与低耦合高内聚原则,构建清晰的组件结构。 定义清晰的模块边界 每个模块应封装特定功能,对外暴…

    好文分享 2025年12月19日
    000
  • 利用C++模板技术提升嵌入式系统性能的实践

    在嵌入式系统开发中,资源受限是常态,性能和内存使用效率至关重要。c++++模板技术不仅支持泛型编程,还能在编译期完成大量逻辑处理,减少运行时开销,从而显著提升系统性能。通过合理使用模板,开发者可以在不牺牲可维护性的前提下,实现高效、可复用的底层代码。 编译期计算与常量优化 模板结合 constexp…

    好文分享 2025年12月19日
    000
  • 在资源受限的嵌入式系统中优化C++内存管理

    在资源受限的嵌入式系统中,c++++内存管理直接影响系统稳定性与性能。由于缺乏虚拟内存、堆空间有限且不能依赖垃圾回收机制,必须从设计和编码层面主动控制内存使用。核心策略包括避免动态分配、预分配内存池、使用轻量级替代标准库组件。 禁用或严格限制动态内存分配 嵌入式环境中,malloc 和 new 可能…

    好文分享 2025年12月19日
    000
  • 嵌入式系统实时任务中使用C++构建稳健调度机制

    在嵌入式系统中实现实时任务调度时,c++++ 提供了比 c 更丰富的抽象能力与类型安全机制,但同时也带来对资源开销和确定性的更高要求。构建一个稳健的实时调度机制,关键在于结合 c++ 的优势,同时规避其可能影响实时性的特性。以下从设计原则、核心组件和实现技巧三个方面展开说明。 调度器设计:基于时间片…

    好文分享 2025年12月19日
    000
  • 嵌入式系统驱动开发中高效应用C++面向对象思想

    在嵌入式系统驱动开发中,很多人认为#%#$#%@%@%$#%$#%#%#$%@_9e6df79f947a44c++8a2ba49c4428632a1是唯一可行的选择,主要出于对资源占用和执行效率的顾虑。但随着mcu性能提升和编译器优化进步,c++的面向对象思想可以在不牺牲性能的前提下,显著提升代码的…

    好文分享 2025年12月19日
    000
  • 嵌入式系统固件开发中采用C++实现可维护性工程

    在嵌入式系统固件开发中,传统上多使用#%#$#%@%@%$#%$#%#%#$%@_9e6df79f947a44c++8a2ba49c4428632a1,因其轻量、高效且与硬件贴近。但随着系统复杂度提升,对代码可维护性、复用性和模块化设计的要求越来越高,采用c++进行固件开发成为一种有效提升工程可维护…

    好文分享 2025年12月19日
    000
  • 将C++与RTOS结合实现嵌入式系统高实时性方案

    在嵌入式系统开发中,实时性是许多关键应用(如工业控制、汽车电子、无人机飞控)的核心需求。c++++ 以其面向对象、代码复用和类型安全等优势,正逐步替代 c 成为嵌入式开发的主流语言。将 c++ 与实时操作系统(rtos)结合,可以在保证高实时性的同时提升软件的可维护性和扩展性。 选择合适的RTOS支…

    好文分享 2025年12月19日
    000
  • 在嵌入式系统通信协议中应用C++实现高效解析

    在嵌入入式系统中,通信协议的解析效率直接影响系统的实时性和资源利用率。虽然传统上多采用#%#$#%@%@%$#%$#%#%#$%@_9e6df79f947a44c++8a2ba49c4428632a1进行开发,但合理使用c++可以在保持高性能的同时提升代码的可维护性与扩展性。通过发挥c++的特性,如…

    好文分享 2025年12月19日
    000
  • 使用C++构建嵌入式系统中的事件驱动框架

    在嵌入式系统中,资源受限和实时性要求高,采用事件驱动架构(event-driven arc++hitecture)可以有效提升系统的响应效率和模块解耦程度。使用c++构建这样的框架,既能利用其面向对象和模板特性增强代码可维护性,又能通过精细控制避免性能开销。以下是实现一个轻量级、适用于嵌入式环境的事…

    好文分享 2025年12月19日
    000
  • 使用C++构建嵌入式系统中的可扩展插件化架构

    在嵌入式系统中,资源受限和实时性要求高是常态。使用c++++构建可扩展的插件化架构,可以在保证性能的同时提升系统的模块化与维护性。关键在于设计清晰的接口、动态加载机制以及低耦合的模块管理方式。 定义统一的插件接口 插件化架构的核心是抽象出稳定、通用的接口。所有插件必须实现该接口,主系统通过接口与插件…

    好文分享 2025年12月19日
    000
  • 在嵌入式系统引导程序中集成高性能C++模块

    在嵌入式系统中,引导程序(bootloader)通常以#%#$#%@%@%$#%$#%#%#$%@_9e6df79f947a44c++8a2ba49c4428632a1编写,强调启动速度、内存控制和硬件直接操作。然而,随着系统复杂度提升,开发者希望在引导阶段引入更高级的逻辑处理能力,比如配置解析、安…

    好文分享 2025年12月19日
    000
  • C++怎么使用ZeroMQ进行消息通信_C++分布式通信与ZeroMQ使用

    答案:在C++中集成ZeroMQ需先创建上下文并初始化套接字,选择REQ/REP或PUB/SUB等模式实现进程间通信,通过send/recv传输二进制安全的多段消息。 在C++项目中集成ZeroMQ进行消息通信,核心在于理解其异步、模式化的通信机制,并正确使用API。它比传统Socket更高级,能快…

    2025年12月19日
    000
  • 通过现代C++特性提升嵌入式系统跨平台能力

    在嵌入式系统开发中,跨平台能力直接影响代码的可移植性、维护成本和开发效率。现代c++++(c++11及以上)引入了许多语言和标准库特性,能够在不依赖外部框架的前提下,显著提升嵌入式代码在不同架构(如arm cortex-m、risc-v、x86等)和编译器(gcc、clang、iar、keil等)间…

    好文分享 2025年12月19日
    000
  • C++怎么处理命令行参数_C++程序输入解析与命令行参数处理方法

    答案:C++命令行参数处理有三种方式:基础argc/argv遍历适用于简单程序;getopt适合Linux环境下的规范选项解析;复杂项目推荐CLI11等库实现声明式解析与自动帮助生成功能。 在C++中处理命令行参数是编写实用工具和系统程序的基础技能。程序启动时,可以通过主函数的参数接收外部输入,进而…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信