重温设计模式--享元模式

news2024/12/24 5:26:19

文章目录

      • 享元模式(Flyweight Pattern)概述
      • 享元模式的结构
      • C++ 代码示例1
      • 应用场景
      • C++示例代码2

享元模式(Flyweight Pattern)概述

  1. 定义
    运用共享技术有效地支持大量细粒度的对象。
    享元模式是一种结构型设计模式,它主要用于减少创建对象的数量,以减少内存占用和提高性能。该模式通过共享尽可能多的对象数据来实现这一点,对于那些在系统中会大量重复出现且内部状态可以被共享的对象,将其可共享的部分提取出来,进行统一管理和共享使用,而把那些依赖于具体场景不能共享的部分作为外部状态传入。

  2. 作用

    • 节省内存:在一些应用场景中,比如游戏开发里有大量相似的游戏元素(如相同外观的小兵角色等),如果每个都创建独立的对象,会消耗大量内存。通过享元模式共享这些元素的通用部分,能大大减少内存开销。
    • 提升性能:创建对象是有一定开销的,当大量重复创建相似对象时,会影响系统性能。享元模式避免了不必要的重复创建,使得系统在运行时更加高效,例如在图形绘制系统中,频繁绘制相同图形时,共享图形对象能加快绘制速度。
    • 便于管理对象状态:将对象的状态分为内部状态(可共享的、不随外部环境变化的部分)和外部状态(随具体场景变化的部分),使得对象状态的管理更加清晰和有条理,便于对不同场景下对象的行为进行控制。

享元模式的结构

  1. 抽象享元(Flyweight)
    这是所有具体享元类的抽象,定义了享元对象的公共接口,通常包含可以接收外部状态并执行相应操作的方法等。

  2. 具体享元(Concrete Flyweight)
    实现了抽象享元的接口,它持有内部状态,并且这些内部状态在不同实例间是可以共享的。具体享元对象的内部状态决定了它的固有行为,在创建后通常不会改变(除非有特定的修改逻辑设计)。

  3. 享元工厂(Flyweight Factory)
    负责创建和管理享元对象。它维护一个享元对象的池(通常可以用std::map等容器来实现),在客户端请求创建享元对象时,先检查池中是否已经存在,如果存在就直接返回已有的对象,避免重复创建;如果不存在,则创建一个新的具体享元对象并放入池中,然后返回给客户端。

  4. 客户端(Client)
    客户端是使用享元对象的地方,它需要维护享元对象的外部状态,并且通过享元对象的接口,传入外部状态来调用相应的操作,完成业务逻辑。

在这里插入图片描述

C++ 代码示例1

以下以一个简单的图形绘制系统为例,假设有多种颜色的圆形需要绘制,颜色是可以共享的内部状态(因为可能有多个圆形都是同一种颜色),而圆形的坐标位置是外部状态(每个圆形在不同位置绘制)。

#include <iostream>
#include <map>
#include <string>

// 抽象享元类
class Shape
{
public:
	virtual void draw(int x, int y) = 0;
	virtual ~Shape() {}
};

// 具体享元类 - 圆形
class Circle : public Shape
{
private:
	std::string color;
public:
	Circle(std::string col) : color(col) {}
	void draw(int x, int y) override 
	{
		std::cout << "在坐标(" << x << ", " << y << ")绘制了一个" << color << "的圆形" << std::endl;
	}
};

// 享元工厂类
class ShapeFactory
{
private:
	std::map<std::string, Shape*> shapeMap;
public:
	Shape* getShape(std::string color) 
	{
		if (shapeMap.find(color) == shapeMap.end()) 
		{
			shapeMap[color] = new Circle(color);
		}
		return shapeMap[color];
	}
	~ShapeFactory()
	{
		for (auto it = shapeMap.begin(); it!= shapeMap.end(); ++it)
		{
			delete it->second;
		}
	}
};

// 客户端代码
int main()
{
	ShapeFactory factory;
	Shape* circle1 = factory.getShape("红色");
	Shape* circle2 = factory.getShape("红色");
	Shape* circle3 = factory.getShape("蓝色");

	circle1->draw(10, 20);
	circle2->draw(30, 40);
	circle3->draw(50, 60);

	return 0;
}

在上述代码中:

  • Shape是抽象享元类,定义了draw方法作为所有图形绘制的统一接口,这里只考虑简单的绘制操作示例,传入坐标参数来表示绘制的位置。

  • Circle是具体享元类,它有一个表示颜色的私有成员变量color作为内部状态,实现了draw方法,根据传入的坐标在相应位置绘制指定颜色的圆形。

  • ShapeFactory是享元工厂类,内部用std::map容器维护了一个形状对象的映射表,getShape方法根据传入的颜色参数来查找是否已经创建过对应的圆形对象,如果没有则创建一个新的Circle对象并放入映射表中,然后返回对应的形状对象,这样就实现了相同颜色的圆形对象的共享,避免重复创建。在其析构函数中,对创建的所有形状对象进行内存释放,防止内存泄漏。

  • main函数作为客户端代码部分,首先创建了享元工厂对象,然后通过工厂获取不同颜色的圆形对象(这里获取了两个红色的圆形和一个蓝色的圆形,其中两个红色圆形其实是共享同一个对象实例),最后分别传入不同的坐标调用draw方法来模拟绘制圆形的操作,展示了享元模式如何在图形绘制场景中通过共享对象来节省内存和管理相似对象的情况。

这个代码示例在VS2010环境下可以正常编译运行,实现了享元模式的基本功能应用,你可以根据实际需求进一步扩展和完善,比如添加更多的图形种类、更复杂的状态管理等内容。

应用场景

  1. 游戏开发
    • 游戏中存在大量重复的游戏元素,比如相同外观的怪物、建筑等。以怪物为例,它们可能具有相同的外观(纹理、模型等可共享的内部状态),只是在游戏地图中的位置(外部状态)不同。通过享元模式,可以让多个相同外观的怪物共享同一个代表外观的对象实例,减少内存占用,同时方便对怪物外观等共享属性进行统一管理和修改。
    • 游戏道具也是如此,很多道具可能有相同的基础样式,但使用时的位置、状态等不同,利用享元模式能优化道具对象的创建和管理。
  2. 图形绘制与可视化系统
    • 在绘制复杂图形界面或者可视化数据展示时,会有大量重复的图形元素,比如绘制地图上的多个相同图标(如代表医院、学校等的图标),这些图标本身的样式(颜色、形状等内部状态)可以共享,只是绘制的坐标位置(外部状态)不一样。享元模式可确保相同样式的图标只创建一次,提升绘制效率并节省内存,方便对图标样式进行统一更新维护。
    • 对于绘制统计图表中的相同类型的图形标记(如柱状图中的柱子样式等),也可以应用享元模式进行优化。
  3. 文本处理与编辑系统
    • 在文字处理软件中,对于字体样式对象,字体的名称、字号、加粗、倾斜等属性是固定的(内部状态),可以共享。而每个文字在文档中的具体位置(外部状态)不同。通过享元模式管理字体样式对象,能避免为每个文字都创建重复的字体样式实例,提高内存利用效率,并且方便对字体样式进行整体的修改和统一管理,比如改变文档中所有某字号字体的颜色等操作。
    • 同样,对于文档中的一些固定格式的段落样式等也可以采用享元模式来优化对象管理,减少内存开销。
  4. 网站开发中的资源管理
    • 网页上经常会有很多重复的图片、图标等资源展示,比如电商网站商品列表中相同类别的商品图片(可能只是商品编号等不同,但图片外观一样),可以把图片资源作为享元对象进行管理,共享相同的图片实例,减少浏览器内存占用,加快网页加载速度,同时便于对图片资源进行更新替换等操作。
    • 网站的一些通用样式组件(如按钮样式、导航栏样式等)也可利用享元模式,将样式相关的内部状态共享,根据页面不同位置等外部状态进行展示和交互,优化网站前端资源的管理和性能。

C++示例代码2

#include<iostream>
#include<list>
#include<string>
#include<vector>
using namespace std;
enum COLOR{RED , BLACK};
//记录名字和坐标
typedef struct NODE
{
	int x;
	int y;
	string name;
}NODE;

class PIECE
{
private:
	COLOR m_color;
public:
	PIECE(COLOR p_color):m_color(p_color){}
	~PIECE(){}
	virtual void  DRAW(){}

};

class REDPIECE:public PIECE
{
public:
	REDPIECE(COLOR p_color):PIECE(p_color){}
	virtual void  DRAW(){cout<<"绘制一颗红棋子"<<endl;}
};

class BLACKPIECE:public PIECE
{
public:
	BLACKPIECE(COLOR p_color):PIECE(p_color){}
	virtual void  DRAW(){cout<<"绘制一颗黑棋子"<<endl;}
};

//棋盘
class BOARD
{
private:
	list<NODE> m_ls;
	PIECE*m_redpiece;
	PIECE*m_blackpiece;
public:
	BOARD()
	{
		m_redpiece=NULL;
		m_blackpiece=NULL;
	}

	~BOARD()
	{
		delete m_blackpiece;
		delete m_redpiece;
	}

	void SetPiece(COLOR p_color , NODE p_node)
	{
		if(p_color==RED)
		{
			if(m_redpiece == NULL)  //有点单例模式的意思
			{
				m_redpiece = new REDPIECE(p_color);	
			}
			cout<<p_node.name<<"在位置("<<p_node.x<<','<<p_node.y<<")";
			m_redpiece->DRAW();
		}
		else
		{
			if(m_redpiece == NULL)  //有点单例模式的意思
			{
				m_redpiece = new REDPIECE(p_color);	
			}
			cout<<p_node.name<<"在位置("<<p_node.x<<','<<p_node.y<<")";
			m_redpiece->DRAW();
		}
		m_ls.push_back(p_node);//这里只存放NODE的信息 ,不用存对象的信息,将会大大减少存储空间的消耗
	}

};


int main()
{
	BOARD *m_board= new BOARD();
	NODE p1;
	p1.name = "马";
	p1.x=1;
	p1.y = 2;


	NODE p2;
	p2.name = "车";
	p2.x=1;
	p2.y = 2;

	REDPIECE *m_red = new REDPIECE(RED);
	m_board->SetPiece(RED , p1);

	BLACKPIECE *m_black = new BLACKPIECE(BLACK);
	m_board->SetPiece(BLACK , p2);

	return 0;

}

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

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

相关文章

Taro小程序开发性能优化实践

我们团队在利用Taro进行秒送频道小程序的同时&#xff0c;一直在探索性能优化的最佳实践。随着需求的不断迭代&#xff0c;项目中的性能问题难免日积月累&#xff0c;逐渐暴露出来影响用户体验。适逢双十一大促&#xff0c;我们趁着这个机会统一进行了Taro性能优化实践&#xf…

纯血鸿蒙APP实战开发——textOverflow长文本省略

介绍 本示例实现了回复评论时&#xff0c;当回复人的昵称与被回复人的昵称长度都过长时&#xff0c;使用textOverflow和maxLines()实现昵称的长文本省略展示的功能。 效果图预览 使用说明 点击评论中的"回复"&#xff0c;在输入框中输入回复内容&#xff0c;点击发…

【java面向对象编程】第九弹----抽象类、接口、内部类

笔上得来终觉浅,绝知此事要躬行 &#x1f525; 个人主页&#xff1a;星云爱编程 &#x1f525; 所属专栏&#xff1a;javase &#x1f337;追光的人&#xff0c;终会万丈光芒 &#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐文章 一、抽象类 1.1基本介绍 &…

Qt笔记:网络编程UDP

一、铺垫 1.Qt框架使用的网络结构的基础就是Linux学习的网络编程基础&#xff1b;所以使用Qt写客户端&#xff0c;使用Linux写服务端&#xff1b;两者是可以实现互联的 二、UDP 网络编程UDP使用套路&#xff1a; 1.首先在.pro文件中加上network&#xff0c;使Qt可以搭载网络…

Redis存在安全漏洞

Redis是美国Redis公司的一套开源的使用ANSI C编写、支持网络、可基于内存亦可持久化的日志型、键值&#xff08;Key-Value&#xff09;存储数据库&#xff0c;并提供多种语言的API。 Redis存在安全漏洞。攻击者利用该漏洞使用特制的Lua脚本触发堆栈缓冲区溢出漏洞&#xff0c;从…

【潜意识Java】蓝桥杯算法有关的动态规划求解背包问题

目录 背包问题简介 问题描述 输入&#xff1a; 输出&#xff1a; 动态规划解法 动态规划状态转移 代码实现 代码解释 动态规划的时间复杂度 例子解析 输出&#xff1a; 总结 作者我蓝桥杯&#xff1a;2023第十四届蓝桥杯国赛C/C大学B组一等奖&#xff0c;所以请听我…

ReactPress 1.6.0:重塑博客体验,引领内容创新

ReactPress 是一个基于Next.js的博客&CMS系统&#xff0c; Github项目地址&#xff1a;https://github.com/fecommunity/reactpress 欢迎Star。 体验地址&#xff1a;http://blog.gaoredu.com/ 今天&#xff0c;我们自豪地宣布ReactPress 1.6.0版本的正式发布&#xff0c;…

单元测试-Unittest框架实践

文章目录 1.Unittest简介1.1 自动化测试用例编写步骤1.2 相关概念1.3 用例编写规则1.4 断言方法 2.示例2.1 业务代码2.2 编写测试用例2.3 生成报告2.3.1 方法12.3.2 方法2 1.Unittest简介 Unittest是Python自带的单元测试框架&#xff0c;适用于&#xff1a;单元测试、Web自动…

带有 Elasticsearch 和 Langchain 的 Agentic RAG

作者&#xff1a;来自 Elastic Han Xiang Choong 讨论并实现 Elastic RAG 的代理流程&#xff0c;其中 LLM 选择调用 Elastic KB。 更多阅读&#xff1a;Elasticsearch&#xff1a;基于 Langchain 的 Elasticsearch Agent 对文档的搜索。 简介 代理是将 LLM 应用于实际用例的…

[react 3种方法] 获取ant组件ref用ts如何定义?

获取ant的轮播图组件, 我用ts如何定义? Strongly Type useRef with ElementRef | Total TypeScript import React, { ElementRef } from react; const lunboRef useRef<ElementRef<typeof Carousel>>(null); <Carousel autoplay ref{lunboRef}> 这样就…

stm32制作CAN适配器5--WinUsb上位机编写

上次我们要stm32制作了一个基于winusb有canfd适配器&#xff0c;今天我们来制作一个上位机程序来进行报文收发。 上位机还是用以前写好的&#xff0c;只是更改下dll文件。 项目链接器&#xff0c;输入&#xff0c;附加依赖项中增加winusb.lib winusb初始化&#xff1a;#incl…

C/C++圣诞树

系列文章 序号直达链接1C/C爱心代码2C/C跳动的爱心3C/C李峋同款跳动的爱心代码4C/C满屏飘字表白代码5C/C大雪纷飞代码6C/C烟花代码7C/C黑客帝国同款字母雨8C/C樱花树代码9C/C奥特曼代码10C/C精美圣诞树11C/C俄罗斯方块12C/C贪吃蛇13C/C孤单又灿烂的神-鬼怪14C/C闪烁的爱心15C…

图解HTTP-HTTP报文

参考资料&#xff1a;图解HTTP HTTP报文 用于HTTP协议交互的信息被称为HTTP报文。请求端的HTTP请求报文&#xff0c;响应端&#xff08;服务器端&#xff09;的叫做响应报文。HTTP报文本身是由多行&#xff08;CR LF作为换行符&#xff09;数据行构成的文本。 请求报文及响…

机器学习基础算法 (一)-线性回归

python 环境的配置参考 从零开始&#xff1a;Python 环境搭建与工具配置 线性回归的 Python 实现 线性回归是一种经典的机器学习算法&#xff0c;用于预测连续的目标变量。它假设目标变量和特征之间存在线性关系。本文将详细介绍线性回归的原理、Python 实现、模型评估和调优&…

Java字符串的|分隔符转List实现方案

字符串处理 问题背景代码实现代码优化原因分析实现方案 注意事项异常处理Maven未识别异常 问题背景 在项目组对账流程中&#xff0c;接收对方系统的对账文件&#xff0c;数据以|为分隔符&#xff0c;读取文件内容&#xff0c;分条入库。 代码实现 Java中将字符串转给list&am…

【HarmonyOs学习日志(14)】计算机网络之域名系统DNS

域名系统DNS 域名系统DNS——从域名解析出IP地址 文章目录 域名系统DNS概述域名到IP地址的解析 互联网的域名结构命名标准 域名服务器域名的解析过程 概述 域名系统DNS&#xff08;Domain Name System&#xff09;是互联网使用的命名系统&#xff0c;用来把便于人们使用的机器…

LabVIEW与PLC点位控制及OPC通讯

在工业自动化中&#xff0c;PLC通过标准协议&#xff08;如Modbus、Ethernet/IP等&#xff09;与OPC Server进行数据交换&#xff0c;LabVIEW作为上位机通过OPC客户端读取PLC的数据并进行监控、控制与处理。通过这种方式&#xff0c;LabVIEW能够实现与PLC的实时通信&#xff0c…

Restaurants WebAPI(四)——Identity

文章目录 项目地址一、Authentication&#xff08;身份认证&#xff09;1.1 配置环境(解决类库包无法引用)1.2 使用Authentication控制Controller的访问1.3 获取User的Context1.3.1 在Application下创建User文件夹1. 创建User.cs record类封装角色信息2. 创建UserContext.cs提供…

Java-32 深入浅出 Spring - IoC 基础 启动IoC 纯注解方式 SpringConfig web.xml

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 大数据篇正在更新&#xff01;https://blog.csdn.net/w776341482/category_12713819.html 目前已经更新到了&#xff1a; MyBatis&#xff…

Android Studio创建新项目并引入第三方so外部aar库驱动NFC读写器读写IC卡

本示例使用设备&#xff1a;https://item.taobao.com/item.htm?spma21dvs.23580594.0.0.52de2c1bbW3AUC&ftt&id615391857885 一、打开Android Studio,点击 File> New>New project 菜单&#xff0c;选择 要创建的项目模版&#xff0c;点击 Next 二、输入项目名称…