C++实现设计模式---享元模式 (Flyweight)

享元模式 (Flyweight)

享元模式 是一种结构型设计模式,它通过共享对象来减少内存使用和对象创建的开销。当系统中有大量相似对象时,享元模式可以避免重复创建相同对象,从而节省内存。


意图

  • 通过共享相同对象来减少内存消耗。
  • 用于系统中存在大量相似对象的场景。

使用场景

  1. 大量相似对象

    • 系统中存在许多共享的对象,这些对象的状态大部分是相同的。
  2. 状态可以分为内部状态和外部状态

    • 内部状态:可以被共享的部分,不会随上下文变化。
    • 外部状态:特定于场景的部分,不可共享,需要由客户端管理。
  3. 节约内存

    • 系统需要大量创建对象,但内存开销成为瓶颈。

参与者角色

  1. 享元接口 (Flyweight)

    • 定义共享对象的公共接口。
  2. 具体享元 (ConcreteFlyweight)

    • 实现享元接口,存储可以共享的内部状态。
  3. 非共享享元 (UnsharedFlyweight)

    • 不可以共享的子类,通常会组合共享的享元对象。
  4. 享元工厂 (FlyweightFactory)

    • 创建并管理享元对象,确保共享对象的唯一性。
  5. 客户端 (Client)

    • 负责管理外部状态,并将外部状态与享元对象结合使用。

示例代码

以下代码展示了如何使用享元模式来共享图形对象,例如圆形。

#include <iostream>
#include <string>
#include <unordered_map>
#include <memory>// 享元接口:图形
class Shape {
public:virtual ~Shape() = default;virtual void draw(const std::string& color) const = 0; // 外部状态为颜色
};// 具体享元类:圆形
class Circle : public Shape {
private:int radius; // 内部状态:半径public:Circle(int r) : radius(r) {}void draw(const std::string& color) const override {std::cout << "Drawing a circle with radius " << radius << " and color " << color << std::endl;}
};// 享元工厂:管理享元对象
class ShapeFactory {
private:std::unordered_map<int, std::shared_ptr<Shape>> shapes; // 缓存共享对象public:std::shared_ptr<Shape> getCircle(int radius) {if (shapes.find(radius) == shapes.end()) {shapes[radius] = std::make_shared<Circle>(radius);std::cout << "Creating a circle with radius " << radius << std::endl;}return shapes[radius];}
};// 客户端代码
int main() {ShapeFactory factory;// 获取共享的圆形对象auto circle1 = factory.getCircle(5);auto circle2 = factory.getCircle(10);auto circle3 = factory.getCircle(5); // 共享对象// 使用享元对象绘制circle1->draw("red");   // 外部状态:红色circle2->draw("blue");  // 外部状态:蓝色circle3->draw("green"); // 外部状态:绿色return 0;
}

代码解析

1. 享元接口 (Shape)
  • 定义了 draw 方法,用于绘制图形,并接收外部状态作为参数:
class Shape {
public:virtual ~Shape() = default;virtual void draw(const std::string& color) const = 0;
};
2. 具体享元类 (Circle)
  • 实现了 Shape 接口,存储可以共享的内部状态(如半径):
class Circle : public Shape {
private:int radius; // 内部状态public:Circle(int r) : radius(r) {}void draw(const std::string& color) const override {std::cout << "Drawing a circle with radius " << radius << " and color " << color << std::endl;}
};
3. 享元工厂 (ShapeFactory)
  • 负责管理享元对象,保证相同的内部状态只创建一个对象:
class ShapeFactory {
private:std::unordered_map<int, std::shared_ptr<Shape>> shapes; // 缓存共享对象public:std::shared_ptr<Shape> getCircle(int radius) {if (shapes.find(radius) == shapes.end()) {shapes[radius] = std::make_shared<Circle>(radius);std::cout << "Creating a circle with radius " << radius << std::endl;}return shapes[radius];}
};
4. 客户端
  • 客户端通过 ShapeFactory 获取共享对象,并将外部状态与内部状态结合:
auto circle1 = factory.getCircle(5); // 创建新对象
auto circle3 = factory.getCircle(5); // 复用已有对象
circle1->draw("red");   // 绘制红色圆形
circle3->draw("green"); // 绘制绿色圆形

优缺点

优点
  1. 减少内存使用
    • 通过共享对象,避免了重复创建相同对象,节省了内存。
  2. 提高系统性能
    • 减少了对象创建的时间和空间开销。
  3. 灵活性高
    • 外部状态与内部状态分离,可以独立管理。
缺点
  1. 复杂性增加
    • 系统需要额外的享元工厂来管理共享对象。
  2. 外部状态的管理困难
    • 客户端需要显式传递外部状态,增加了使用复杂性。

适用场景

  1. 系统有大量相似对象

    • 如文字处理系统中的字符对象,每个字符对象可以共享其字体、大小等内部状态。
  2. 内存消耗成为瓶颈

    • 系统中对象数量庞大,通过共享来减少内存占用。
  3. 对象的状态可以分为内部状态和外部状态

    • 内部状态可以共享,外部状态由客户端管理。

总结

享元模式通过共享相似对象来减少内存使用,是一种优化性能的重要模式。它适用于对象数量庞大且状态大部分可以共享的场景。

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

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

相关文章

K8S 亲和性与反亲和性 深度好文

今天我们来实验 pod 亲和性。官网描述如下&#xff1a; 假设有如下三个节点的 K8S 集群&#xff1a; k8s31master 是控制节点 k8s31node1、k8s31node2 是工作节点 容器运行时是 containerd 一、镜像准备 1.1、镜像拉取 docker pull tomcat:8.5-jre8-alpine docker pull nginx…

Nginx三种不同类型的虚拟主机(基于域名、IP 和端口)

&#x1f3e1;作者主页&#xff1a;点击&#xff01; Nginx-从零开始的服务器之旅专栏&#xff1a;点击&#xff01; &#x1f427;Linux高级管理防护和群集专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2025年1月15日13点14分 目录 1. 基于域名的虚拟主机 …

解析OVN架构及其在OpenStack中的集成

引言 随着云计算技术的发展&#xff0c;虚拟化网络成为云平台不可或缺的一部分。为了更好地管理和控制虚拟网络&#xff0c;Open Virtual Network (OVN) 应运而生。作为Open vSwitch (OVS) 的扩展&#xff0c;OVN 提供了对虚拟网络抽象的支持&#xff0c;使得大规模部署和管理…

【Flink系列】9. Flink容错机制

9. 容错机制 在Flink中&#xff0c;有一套完整的容错机制来保证故障后的恢复&#xff0c;其中最重要的就是检查点。 9.1 检查点&#xff08;Checkpoint&#xff09; 9.1.1 检查点的保存 1&#xff09;周期性的触发保存 “随时存档”确实恢复起来方便&#xff0c;可是需要我…

《Keras 3 在 TPU 上的肺炎分类》

Keras 3 在 TPU 上的肺炎分类 作者&#xff1a;Amy MiHyun Jang创建日期&#xff1a;2020/07/28最后修改时间&#xff1a;2024/02/12描述&#xff1a;TPU 上的医学图像分类。 &#xff08;i&#xff09; 此示例使用 Keras 3 在 Colab 中查看 GitHub 源 简介 设置 本教程将介…

openssl s_server源码剥离

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github&#xff1a;codetoys&#xff0c;所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的&#xff0c;可以在任何平台上使用。 源码指引&#xff1a;github源…

51单片机 DS18B20温度储传感器

DS18B20温度传感器 64-BITROM&#xff1a;作为器件地址&#xff0c;用于总线通信的寻址&#xff0c;是唯一的&#xff0c;不可更改 SCRATCHPAD&#xff08;暂存器&#xff09;&#xff1a;用于总线的数据交互 EEPROM&#xff1a;用于保存温度触发阈值和配置参数 暂存器 单总线…

[操作系统] 深入理解操作系统的概念及定位

概念 任何计算机系统都包含⼀个基本的程序集合&#xff0c;称为操作系统(OS)。 其核心功能如图片所示&#xff0c;包括&#xff1a; 内核 (Kernel)&#xff1a; 内核是操作系统的核心部分&#xff0c;被认为是狭义上的操作系统&#xff0c;直接与硬件打交道。负责进程管理、内…

Java并发编程——线程池(基础,使用,拒绝策略,命名,提交方式,状态)

我是一个计算机专业研0的学生卡蒙Camel&#x1f42b;&#x1f42b;&#x1f42b;&#xff08;刚保研&#xff09; 记录每天学习过程&#xff08;主要学习Java、python、人工智能&#xff09;&#xff0c;总结知识点&#xff08;内容来自&#xff1a;自我总结网上借鉴&#xff0…

nginx 配置代理,根据 不同的请求头进行转发至不同的代理

解决场景&#xff1a;下载发票的版式文件&#xff0c;第三方返回的是url链接地址&#xff0c;但是服务是部署在内网环境&#xff0c;无法访问互联网进行下载。此时需要进行走反向代理出去&#xff0c;如果按照已有套路&#xff0c;就是根据不同的访问前缀&#xff0c;跳转不同的…

docker在不删除容器的情况下修改端口映射

注意&#xff1a;必须先停止docker服务&#xff01;&#xff01;&#xff01;&#xff01; 1) 停止容器 2) 停止docker服务(systemctl stop docker) 3) 修改这个容器的hostconfig.json和config.v2.json文件中的端口 先查看容器id docker inspect jenkins 进入该目录 hostcon…

【js进阶】设计模式之单例模式的几种声明方式

单例模式&#xff0c;简言之就是一个类无论实例化多少次&#xff0c;最终都是同一个对象 原生js的几个辅助方式的实现 手写forEch,map,filter Array.prototype.MyForEach function (callback) {for (let i 0; i < this.length; i) {callback(this[i], i, this);} };con…

专题 - STM32

基础 基础知识 STM所有产品线&#xff08;列举型号&#xff09;&#xff1a; STM产品的3内核架构&#xff08;列举ARM芯片架构&#xff09;&#xff1a; STM32的3开发方式&#xff1a; STM32的5开发工具和套件&#xff1a; 若要在电脑上直接硬件级调试STM32设备&#xff0c;则…

-bash: /java: cannot execute binary file

在linux安装jdk报错 -bash: /java: cannot execute binary file 原因是jdk安装包和linux的不一致 程序员的面试宝典&#xff0c;一个免费的刷题平台

【MySQL】使用C语言链接

&#x1f308; 个人主页&#xff1a;Zfox_ &#x1f525; 系列专栏&#xff1a;MySQL 目录 一&#xff1a;&#x1f525; MySQL connect &#x1f98b; Connector / C 使用&#x1f98b; mysql 接口介绍&#x1f98b; 完整代码样例 二&#xff1a;&#x1f525; 共勉 一&#…

平滑算法 效果比较

目录 高斯平滑 效果对比 移动平均效果比较: 高斯平滑 效果对比 右边两个参数是1.5 2 代码: smooth_demo.py import numpy as np import cv2 from scipy.ndimage import gaussian_filter1ddef gaussian_smooth_array(arr, sigma):smoothed_arr = gaussian_filter1d(arr, s…

P3数据结构、数据类型、数字编码、字符编码:保姆级图文详解

文章目录 前言1、数据结构分类1.1、逻辑结构&#xff1a;线性与非线性1.2、物理结构&#xff1a;连续与分散1.3、数据结构的实现方式1.4、数据结构的选择依据 2、基本数据类型2.1、定义与分类2.2、存储形式 3、数字编码3.1、原码、反码与补码3.2、浮点数编码3.3、整数与浮点数区…

解密AIGC三大核心算法:GAN、Transformer、Diffusion Models原理与应用

在当今数字化时代&#xff0c;人工智能生成内容&#xff08;AIGC&#xff09;技术正以前所未有的速度改变着我们的生活和工作方式。从创意无限的文本生成&#xff0c;到栩栩如生的图像创作&#xff0c;再到动听的音乐旋律&#xff0c;AIGC的魔力无处不在。而这一切的背后&#…

Web前端------HTML表格

一.表格标签介绍 表格&#xff0c;类似操作的软件excel一样&#xff0c;通过规范的行列方式展示数据的一种视图&#xff01; 网页中&#xff08;初级开发&#xff09;&#xff0c;对于这种规范的数据&#xff0c;使用表格标签最方便的&#xff1b; 实际开发&#xff08;高级开…

nginx 配置域名前缀访问 react 项目

说明一下&#xff1a;我是使用域名转发访问的&#xff0c;访问流程如下&#xff1a; 浏览器 》 服务器1 》 服务器2 由于服务器1已经为 https 的访问方式做了 ssl 证书等相关配置&#xff0c;然后转发到服务器2&#xff0c; 所以在服务器2中不需要再配置 ssl 证书相关的东西了&…