C++设计模式——Flyweight享元模式

一,享元模式简介

享元模式是一种结构型设计模式,它将每个对象中各自保存一份数据的方式改为多个对象共享同一份数据,该模式可以有效减少应用程序的内存占用。
享元模式的核心思想是共享和复用,通过设置共享资源来避免创建过多的实例。
当应用程序的内部包含大量的对象,且对象之间包含相似的数据或状态时,可以使用享元模式来共享这些数据或状态。
享元模式的内部涉及到工厂模式的使用,因为它需要创建一个享元工厂来管理共享资源池。这个共享资源池,又称为享元池(Flyweight Pool),这里面包含多个访问共享数据的享元对象。当客户端需要使用一个享元对象时,享元工厂会从池中获取一个已有的享元对象,如果对象不存在,则创建一个新的享元对象。

二,享元模式的结构

1.内部状态(Intrinsic State):对象之间容易重复的、可以共享的、且变动很少的成员变量,该变量在享元模式中被共享。
2.外部状态(Extrinsic State):对象之间各不相同的、不能共享的、且随着不同场景而变化的成员变量,该变量被调用的客户端所设置和更改。
3.享元工厂类(Flyweight Factory):替外部客户端管理共享资源的类。
4.抽象享元类(Flyweight):享元模式的核心,由享元工厂进行创建和管理,里面包含了内部状态,但不包含外部状态。
5.共享的具体享元类(Concrete Flyweight):实现了Flyweight声明的接口并访问和存储了内部状态。
对应UML类图:

代码实现:

#include <iostream>
#include <memory>
#include<unordered_map>using namespace std;class Flyweight {
protected:int id; //内部状态public:Flyweight(int id) : id(id) {}virtual void operation() const = 0;
};class ConcreteFlyweightA : public Flyweight {
public:ConcreteFlyweightA() : Flyweight(1) {}void operation() const override {cout << "Concrete Flyweight A, id: " << id << endl;}
};class ConcreteFlyweightB : public Flyweight {
public:ConcreteFlyweightB() : Flyweight(2) {}void operation() const override {cout << "Concrete Flyweight B, id: " << id << endl;}
};// 定义享元工厂
class FlyweightFactory {
private:std::unordered_map<int, shared_ptr<Flyweight>> flyweights;public:FlyweightFactory() {}//返回享元对象std::shared_ptr<Flyweight> getConcreteFlyweight(int id) {if (flyweights.find(id) == flyweights.end()) {if (id % 2 == 0) {flyweights[id] = make_shared<ConcreteFlyweightA>();}else {flyweights[id] = make_shared<ConcreteFlyweightB>();}}return flyweights[id];}
};int main() {FlyweightFactory factory;shared_ptr<Flyweight> f1 = factory.getConcreteFlyweight(1);shared_ptr<Flyweight> f2 = factory.getConcreteFlyweight(2);shared_ptr<Flyweight> f3 = factory.getConcreteFlyweight(3);f1->operation();f2->operation();f3->operation();return 0;
}

运行结果:

Concrete Flyweight B, id: 2
Concrete Flyweight A, id: 1
Concrete Flyweight B, id: 2

三,享元模式的工作步骤

1.拆分类的成员变量,将成员变量拆分成以下两种:不变的、可能在对象之间重复使用的。变化的、随着应用场景而改动的。
2.将不变的,可重复的成员变量的属性设置为不可修改,并在构造函数中赋初始值。
3.创建享元类,并将共享的成员变量集成到享元类。
4.创建享元工厂类来管理共享的资源池,客户端与享元对象的交互借助享元工厂来实现。
5.优化共享资源池的代码实现,这可能涉及到事件驱动、回调函数或者策略模式的应用。

四,享元模式的应用场景

图形或图像处理:在大型游戏或图形编辑器开发中,同一个形状(如矩形)或颜色等状态会重复出现很多次,基于享元模式可以降低内存开销。
数据库处理优化:数据库被频繁地连接和请求时,享元模式可以管理这些连接并复用它们,提高处理的性能。
UI组件开发:在用户界面中,当创建多个界面窗口时,像按钮、图标等小部件会在创建界面窗口时有大量重复,使用享元模式可以减少界面之间重复组件的数量,提高性能。

五,享元模式的优缺点

享元模式的优点:
1.增加了系统资源的可重用性,节省了系统资源。
2.基于共享的结构,降低了内存消耗。
3.系统可扩展性强,新增对象时可直接复用共享资源。
4.降低了对象内部的结构复杂性。
享元模式的缺点:
1.使代码结构更加复杂。
2.当需要被共享的资源量级很小时,该模式的性能提升并不显著。
3.将共享变量放在构造函数中进行赋值,额外增加了初始化的时间。
4.引入了"共享"这种结构,会导致潜在的线程安全问题。
5.编写代码需要考虑保证状态的同步和一致性问题,否则会导致bug的产生。

六,代码实战

Demo:模拟字符编辑器
#include <iostream>
#include <map>using namespace std;//Flyweight
class Character {
public:Character(char symbol) : symbol_(symbol) {}Character() = default;void print() const {cout << "Character: " << symbol_ << endl;}private:char symbol_;
};//Flyweight factory
class CharacterFactory {
public:static const Character& getCharacter(char symbol) {if (characters_.find(symbol) == characters_.end()) {characters_[symbol] = Character(symbol);cout << "Create new char." << endl;}return characters_[symbol];}private:static map<char, Character> characters_;
};map<char, Character> CharacterFactory::characters_;int main() {const Character& A = CharacterFactory::getCharacter('A');const Character& B = CharacterFactory::getCharacter('B');//Reusing 'A'const Character& A2 = CharacterFactory::getCharacter('A');A.print();B.print();A2.print();return 0;
}

运行结果:

Create new char.
Create new char.
Character: A
Character: B
Character: A

七,参考阅读

https://softwarepatterns.com/cpp/flyweight-software-pattern-cpp-example
https://www.pentalog.com/blog/design-patterns/flyweight-design-patterns/
https://design-patterns.readthedocs.io/zh-cn/latest/structural_patterns/flyweight.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/32845.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MSPM0G3507——定时器例程1——TIMA_periodic_repeat_count

以下示例以周期模式配置TimerA0&#xff0c;并使用重复计数功能每隔2秒切换一次GPIO。注意&#xff1a;重复计数功能特定于TimerA0实例&#xff0c;而不是其他TimerA实例。这里是一次500毫秒&#xff0c;重复了四次 主函数&#xff1a; #include "ti_msp_dl_config.h&quo…

20240621日志:大模型压缩-从闭源大模型蒸馏

目录 1. 核心内容2. 方法2.1 先验估计2.2 后验估计2.3 目标函数 3. 交叉熵损失函数与Kullback-Leibler&#xff08;KL&#xff09;损失函数 location&#xff1a;beijing 涉及知识&#xff1a;大模型压缩、知识蒸馏 Fig. 1 大模型压缩-知识蒸馏 1. 核心内容 本文提出在一个贝…

Program-of-Thoughts(PoT):结合Python工具和CoT提升大语言模型数学推理能力

Program of Thoughts Prompting:Disentangling Computation from Reasoning for Numerical Reasoning Tasks github&#xff1a;https://github.com/wenhuchen/Program-of-Thoughts 一、动机 数学运算和金融方面都涉及算术推理。先前方法采用监督训练的形式&#xff0c;但这…

发表在SIGMOD 2024上的高维向量检索/向量数据库/ANNS相关论文

前言 SIGMOD 2024会议最近刚在智利圣地亚哥结束&#xff0c;有关高维向量检索/向量数据库/ANNS的论文主要有5篇&#xff0c;涉及混合查询&#xff08;带属性或范围过滤的向量检索&#xff09;优化、severless向量数据库优化、量化编码优化、磁盘图索引优化。此外&#xff0c;也…

微信小程序入门2

微信开发者工具的安装方法 1.打开微信开发者工具下载页面 在微信小程序管理后台的左侧边栏中选择“开发工具”&#xff0c;然后选择“开发者工具”&#xff0c;即可找到微信开发者工具的下载页面。 2.打开微信开发者工具的下载链接页面 单击“下载” 按钮下载&#xff0c;即…

越复杂的CoT越有效吗?Complexity-Based Prompting for Multi-step Reasoning

Complexity-Based Prompting for Multi-step Reasoning 论文&#xff1a;https://openreview.net/pdf?idyf1icZHC-l9 Github&#xff1a;https://github.com/FranxYao/chain-of-thought-hub 发表位置&#xff1a;ICLR 2023 Complexity-Based Prompting for Multi-step Reason…

STM32 - LED灯 蜂鸣器

&#x1f6a9; WRITE IN FRONT &#x1f6a9; &#x1f50e; 介绍&#xff1a;"謓泽"正在路上朝着"攻城狮"方向"前进四" &#x1f50e;&#x1f3c5; 荣誉&#xff1a;2021|2022年度博客之星物联网与嵌入式开发TOP5|TOP4、2021|2222年获评…

Pytest框架中pytest.mark功能

文章目录 mark功能 1. 使用pytest.mark.skip 2. 使用pytest.mark.skipif 3. 使用 pytest.mark.xfail 4使用pytest.mark.parametrize 5 使用pytest.mark.自定义标记 6 使用pytest.mark.usefixtures pytest 的mark功能在pytest官方文档是这样解释的&#xff1a; https://…

stm32学习笔记---GPIO输出(代码部分)LED闪烁/流水灯/蜂鸣器

目录 面包板的使用方法 第一个演示代码&#xff1a;LED闪烁 最后一次快速新建工程演示 点击新建工程 选择芯片 在工程文件夹中创建Start、Library、User Start文件夹的必备文件复制操作 Library文件夹的必备文件复制操作 User文件夹的必备文件复制操作 在keil中创建S…

关于数据登记的六点观察|数据与治理思享会(第1期)圆满举行

本文内容转载自 数据与治理专委会。 鼹鼠哥有幸在上周参与了数据大讲堂的首次线下活动&#xff0c;也做了个简短笔记 [最新]清华数据大讲堂线下思享会 因为上次是个人笔记&#xff0c;有些内容不方便些。既然今天官方公众号发出来了&#xff0c;就在这里把官方的内容也给大家转…

Repair LED lights

Repair LED lights 修理LED灯&#xff0c;现在基本用灯带&#xff0c;就是小型LED灯串联一起的 1&#xff09;拆旧灯条&#xff0c;这个旧的是用螺丝拧的产品 电闸关掉。 2&#xff09;五金店买一个&#xff0c;这种是磁铁吸附的产品 现在好多都是铝线啊。。。 小部件&#x…

【大数据离线项目四:什么是海豚调度?怎么使用可以将海豚调度应用到我们的大数据项目开发中?】

前言&#xff1a; &#x1f49e;&#x1f49e;大家好&#xff0c;我是书生♡&#xff0c;今天主要和大家分享一下什么是海豚调度&#xff1f;怎么使用可以将海豚调度应用到我们的项目开发中&#xff1f;希望对大家有所帮助。 &#x1f49e;&#x1f49e;代码是你的画笔&#xf…

数组 (java)

文章目录 一维数组静态初始化动态初始化 二维数组静态初始化动态初始化 数组参数传递可变参数关于 main 方法的形参 argsArray 工具类sort 中的 comparable 和 comparatorcomparator 比较器排序comparable 自然排序 一维数组 线性结构 静态初始化 第一种&#xff1a;int[] a…

IDEA插件推荐-CodeGeex

功能&#xff1a;这个插件可以实现快速翻译代码&#xff0c;json文件格式转换&#xff0c;代码语言类型转换。 安装方式&#xff1a;File->Settings->Plugins->MarketPlace->搜索“CodeGeex”即可 &#xff08;CodeGeex功能展示&#xff09; &#xff08;CodeGeex…

模拟算法讲解

模拟算法是一种基于实际情况模拟的算法&#xff0c;通过模拟现实世界中的系统或过程&#xff0c;来研究它们的性质和行为。模拟算法可以用于解决各种问题&#xff0c;包括物理模拟、经济模拟、社会模拟等。 模拟算法的基本步骤包括&#xff1a; 定义问题&#xff1a;明确需要模…

【STM32c8t6】AHT20温湿度采集

【STM32c8t6】AHT20温湿度采集 一、探究目的二、探究原理2.1 I2C2.1. 硬件I2C2.1. 软件I2C 2.2 AHT20数据手册 三、实验过程3.1 CubeMX配置3.2 实物接线图3.3 完整代码3.4 效果展示 四、探究总结 一、探究目的 学习I2C总线通信协议&#xff0c;使用STM32F103完成基于I2C协议的A…

android串口助手apk下载 源码 演示 支持android 4-14及以上

android串口助手apk下载 1、自动获取串口列表 2、打开串口就开始接收 3、收发 字符或16进制 4、默认发送at\r\n 5、android串口助手apk 支持android 4-14 &#xff08;Google seral port 太老&#xff09; 源码找我 需要 用adb root 再setenforce 0进入SELinux 模式 才有权限…

【数据结构导论】自考笔试题:伪代码练习题汇总 1

目录 一、开源项目推荐 二、线性表的基本运算在单链表上的实现 &#xff08;1&#xff09;初始化 &#xff08;2&#xff09;插入 p 指向的新结点的操作 &#xff08;3&#xff09;删除 *p 节点 三、循环链表 &#xff08;1&#xff09;在单链表中 &#xff08;2&…

如何打包数据库文件

使用 mysqldump 命令&#xff1a; mysqldump -u username -p database_name > output_file.sql username 是数据库的用户名。database_name 是要导出的数据库名称。output_file.sql 是导出的 SQL 文件名&#xff0c;可以自定义。 示例&#xff1a; mysqldump -u root -p…

urfread刷算法题day1|LeetCode2748.美丽下标的数目

题目 题目链接 LeetCode2748.美丽下标对的数目 题目描述 给你一个下标从 0 开始的整数数组 nums 。 如果下标对 i、j 满足 0 ≤ i < j < nums.length &#xff0c; 如果 nums[i] 的 第一个数字 和 nums[j] 的 最后一个数字 互质 &#xff0c; 则认为 nums[i] 和 nums…