二十三种设计模式-享元模式
享元模式(Flyweight Pattern)是一种结构型设计模式,旨在通过共享相同对象来减少内存使用,尤其适合在大量重复对象的情况下。
核心概念
享元模式的核心思想是将对象的**可共享部分(内部状态)提取出来进行共享,而将不可共享部分(外部状态)**通过参数传递。这种方式可以显著减少内存占用,提高性能。
角色组成
-
抽象享元(Flyweight)
定义了享元对象的接口,包含内部状态和外部状态的管理方法。 -
具体享元(Concrete Flyweight)
实现了抽象享元接口,存储和管理内部状态。 -
享元工厂(Flyweight Factory)
负责创建和管理享元对象,确保相同内部状态的对象只被创建一次。 -
非共享享元(Unshared Concrete Flyweight)
不共享的部分,存储每个对象独立的状态。
应用场景
享元模式适用于以下场景:
-
大量相似对象的创建:当系统中存在大量相似对象,且这些对象占用大量内存时。
-
对象的共享性较强:对象的某些部分可以共享(如颜色、字体),而某些部分是特有的(如位置、大小)。
-
缓存场景:如数据库连接池、线程池等,通过共享技术减少内存占用。
优点
-
节省内存:通过共享对象,减少了内存中对象的数量。
-
提高性能:减少了对象创建和销毁的开销。
-
易于扩展:新的享元对象可以方便地加入系统。
缺点
-
实现复杂:需要维护共享对象的管理机制,增加了系统的复杂性。
-
线程安全问题:在多线程环境下,需要确保共享对象的线程安全。
实现示例
以下是一个简单的Java实现示例:
// 抽象享元类
interface Flyweight {void operation(String extrinsicState);
}// 具体享元类
class ConcreteFlyweight implements Flyweight {private String intrinsicState;public ConcreteFlyweight(String intrinsicState) {this.intrinsicState = intrinsicState;}@Overridepublic void operation(String extrinsicState) {System.out.println("Intrinsic State: " + intrinsicState + ", Extrinsic State: " + extrinsicState);}
}// 享元工厂类
class FlyweightFactory {private Map<String, Flyweight> flyweights = new HashMap<>();public Flyweight getFlyweight(String intrinsicState) {if (!flyweights.containsKey(intrinsicState)) {flyweights.put(intrinsicState, new ConcreteFlyweight(intrinsicState));}return flyweights.get(intrinsicState);}
}// 客户端代码
public class Client {public static void main(String[] args) {FlyweightFactory factory = new FlyweightFactory();Flyweight flyweight1 = factory.getFlyweight("A");Flyweight flyweight2 = factory.getFlyweight("B");Flyweight flyweight3 = factory.getFlyweight("A");flyweight1.operation("ClientState1");flyweight2.operation("ClientState2");flyweight3.operation("ClientState3");}
}
运行结果:
Intrinsic State: A, Extrinsic State: ClientState1
Intrinsic State: B, Extrinsic State: ClientState2
Intrinsic State: A, Extrinsic State: ClientState3
从输出可以看出,享元模式成功地实现了对相同对象的复用。
注意事项
-
合理划分内部状态和外部状态:内部状态存储在对象内部,外部状态通过参数传递。
-
享元对象的不可变性:享元对象应为不可变对象,避免线程安全问题。
-
权衡利弊:享元模式虽然可以节省内存,但会增加系统的复杂性。需要根据具体场景权衡是否使用。
享元模式在实际项目中广泛应用于需要优化内存使用和提高性能的场景,如缓存系统、图形界面、文本编辑器等。
享元模式(Flyweight Pattern)适用于以下具体场景:
1. 大量相似对象的场景
当系统中存在大量相似对象时,可以使用享元模式来减少内存占用。例如:
-
文本编辑器中的字符对象:文本编辑器中大量重复出现的字符对象可以使用享元模式来共享。每个字符对象的字体、大小等内部状态可以共享,而位置、颜色等外部状态则由客户端管理。
-
围棋棋子:在围棋程序中,黑色和白色棋子可以共享,而棋子的位置则作为外部状态。
-
象棋棋子:象棋程序中,棋子的颜色(黑色或白色)可以共享,而棋子的位置则作为外部状态。
2. 缓存场景
享元模式可以用于实现缓存,特别是在需要频繁访问、计算成本高昂的数据时。通过共享已计算好的数据,可以提高系统的性能。
3. 数据库连接池和线程池
在数据库连接池和线程池中,享元模式可以用于共享可用的连接或线程对象,避免频繁地创建和销毁,从而提高系统性能。
4. 图形处理和游戏开发
-
图形对象共享:在图形处理中,例如绘制大量相似的圆形、矩形等图形对象时,可以通过享元模式共享图形的内部状态(如颜色、大小等),而将位置等外部状态传递给具体对象。
-
在线游戏中的角色属性:游戏中大量玩家角色的不变属性(如名称、等级、经验值等)可以共享,而装备、技能等级等可变属性则单独存储。
5. 大数据处理
在处理大量数据时,可能会存在大量的重复对象,如图像处理中的像素点、文本处理中的单词等。通过享元模式可以减少内存消耗和提高处理速度。
6. 性能和内存瓶颈问题
当应用程序遇到性能瓶颈,尤其是在内存使用上,享元模式可以通过复用对象来减轻内存压力。
7. 字符串常量池
Java中的字符串常量池是享元模式的一个经典应用。字符串常量池通过共享相同的字符串对象,减少了内存占用。
总结
享元模式适用于需要优化内存使用和提高性能的场景,特别是在系统中存在大量相似对象、对象创建和销毁成本较高、以及需要频繁访问和共享数据的情况下。然而,享元模式也会增加系统的复杂性,需要合理权衡其优缺点。
相关文章:
二十三种设计模式-享元模式
享元模式(Flyweight Pattern)是一种结构型设计模式,旨在通过共享相同对象来减少内存使用,尤其适合在大量重复对象的情况下。 核心概念 享元模式的核心思想是将对象的**可共享部分(内部状态)提取出来进行共…...
前端【10】jQuery DOM 操作
目录 jquery捕获查取 获得内容 - text()、html() 以及 val() 获取属性 - attr() 编辑 jQuery 修改/设置内容和属性 设置内容 - text()、html() 以及 val() 设置属性 - attr() jQuery添加元素 jQuery - 删除元素 前端【9】初识jQuery:让JavaScript变得更简…...
Day34:字符串的替换
在 Python 中,字符串替换是一个非常常见的操作,主要用于修改字符串中的某些部分。字符串的替换操作通常不修改原始字符串,因为字符串在 Python 中是不可变的,而是返回一个新的字符串。 Python 提供了 str.replace() 方法来执行替…...
汇编的使用总结
一、汇编的组成 1、汇编指令(指令集) 数据处理指令: 数据搬移指令 数据移位指令 位运算指令 算术运算指令 比较指令 跳转指令 内存读写指令 状态寄存器传送指令 异常产生指令等 2、伪指令 不是汇编指令,但是可以起到指令的作用,伪…...
力扣【347. 前 K 个高频元素】Java题解(堆)
TopK问题,我们直接上堆。 首先遍历一次然后把各个数字的出现频率存放在哈希表中便于后面堆的操作。 因为是出现频率前 k 高,所以用小顶堆,当我们遍历的频率值大于堆顶值时就可以替换堆顶。 代码: class Solution {public int[] …...
【计算机网络】host文件
host文件的主要功能: 域名解析 本地映射:host文件的主要功能是将**域名映射到相应的 IP 地址**。当计算机需要访问一个网站或服务时,它会首先在 host文件中查找该域名对应的 IP 地址。如果在 host文件中找到了匹配的域名和 IP 地址映射&…...
git如何设置pull的时候有些文件不pull
在 Git 中,没有直接的方法在 git pull 时排除特定文件,但可以通过以下方式实现类似效果: 方法 1: 使用 .gitignore .gitignore 文件可以忽略未跟踪的文件,但对已跟踪的文件无效。如果你希望某些文件不被拉取,可以先将…...
第五节 MATLAB命令
本节的内容将提供常用的一些MATLAB命令。 在之前的篇章中我们已经知道了MATLAB数值计算和数据可视化是一个交互式程序,在它的命令窗口中您可以在MATLAB提示符“>>”下键入命令。 MATLAB管理会话的命令 MATLAB提供管理会话的各种命令。如下表所示:…...
性能测试丨JVM 性能数据采集
什么是JVM性能数据采集? JVM性能数据采集是指通过一些工具和技术采集与Java虚拟机相关的性能数据。这些数据包括但不限于内存使用、CPU使用、垃圾回收(GC)行为、线程活动等。合理地分析这些数据,可以帮助我们找出系统的瓶颈&…...
vue3 vue2区别
Vue 3 和 Vue 2 之间存在多个方面的区别,以下是一些主要的差异点: 1. 性能改进 Vue 3:在性能上有显著提升,包括更小的包体积、更快的渲染速度和更好的内存管理。Vue 2:性能相对较低,尤其是在大型应用中。…...
快速更改WampServer根目录php脚本
快速更改WampServer根目录php脚本 <?php // 配置文件地址 $apacheConfPath C:\Install\CTF\Wampserver\bin\apache\apache2.4.62.1\conf\httpd.conf; $apacheConfPath2 C:\Install\CTF\Wampserver\bin\apache\apache2.4.62.1\conf\extra\httpd-vhosts.conf; // 新根目录…...
金价高企需求疲软,周大福近三个月零售值下降14.2%
1月22日,周大福在港交所发布公告称,截至2024年12月31日止三个月(第三季度),集团零售值下降14.2%。其中,中国内地市场零售值下降13.0%,香港及澳门市场零售值下降20.4%,其他市场零售值…...
想品客老师的第七天:闭包和作用域
闭包之前的内容写在这里 环境、作用域、回收 首先还是数据的回收问题,全局变量一般都是通过关闭页面回收的;而局部变量的值不用了,会被自动回收掉 像这种写在全局里的就不会被主动回收捏: let title 荷叶饭function fn() {ale…...
Web 代理、爬行器和爬虫
目录 Web 在线网页代理服务器的使用方法Web 在线网页代理服务器使用流程详解注意事项 Web 请求和响应中的代理方式Web 开发中的请求方法借助代理进行文件下载的示例 Web 服务器请求代理方式代理、网关和隧道的概念参考文献说明 爬虫的工作原理及案例网络爬虫概述爬虫工作原理 W…...
速通Docker === Docker 镜像分层存储机制
目录 分层存储的概念 分层存储的实现 镜像层 容器层 分层存储的优势 1. 镜像轻量化 2. 快速构建与部署 3. 高效的镜像共享 4. 版本控制 分层存储的示例 容器层的临时性与数据持久化 总结 Docker 的分层存储机制是其核心特性之一,它使得镜像的构建、共享和…...
开源智慧园区管理系统对比五款主流产品探索智能运营新模式
内容概要 在这个数字化迅速发展的时代,园区管理也迎来了全新的机遇和挑战。众所周知,开源智慧园区管理系统作为一种创新解决方案,正逐步打破传统管理的局限性。它的开放性不仅使得系统可以根据具体需求进行灵活调整,也为用户提供…...
用Devc++与easyx一步一步做游戏[启动界面部分]-解决hover闪烁问题及优化
在之前的博文中《用Devc与easyx一步一步做游戏[启动界面部分]-之按钮制作》,我们利用Devc和easyx完成了游戏启动界面按钮的基本制作,实现了按钮的绘制以及鼠标悬停时的信息提示功能。然而,目前还存在一个问题,即鼠标移动时&#x…...
非根目录部署 nextjs 项目,资源文件 请求404 的问题
最近在学习next项目编写的代码放到服务器上静态资源404 先分析问题 到服务器上查看是有资源目录的是不是项目配置有问题是不是nginx配置有问题 经过排查1和2是没有问题的目前来看只有3 检查一下nginx配置 尝试着把静态资源的配置禁用 问题解决 我的next项目用的是pm2管理…...
Kafka常见问题之 `javax.management.InstanceAlreadyExistsException`
文章目录 Kafka常见问题之 javax.management.InstanceAlreadyExistsException1. 概述2. 常见原因3. 具体异常示例4. 解决方案4.1 确保单一 Kafka Producer 实例4.2 配置 Kafka Broker 和 Producer 使用唯一的 JMX 名称(对于Producer重点检查 client.id)4…...
mysql 学习5 mysql图形化界面DataGrip下载 安装 使用
一、下载和安装 下载: 其他版本 - DataGrip PS:安装目录最好不要有中文。 C:\Program Files\JetBrains\DataGrip 2023.3.4 二、 夸克网盘分享 当前电脑是下载到 :D:\Ctool\mysql\datagrip2023_3_4\datagrip2024\jetbra-datagrip\scripts …...
读写和解析简单的 nc 文件
NetCDF 文件格式在气象数据工程领域占据着举足轻重的地位,其结构灵活、强兼容性等优势使其成为该领域的一个标准。无论是从事学术研究还是工程实践,掌握这种数据格式变得越发重要。其次,我注意到目前社区中气象编程大多数课程都聚焦于某个特定…...
LLM:BERT or BART 之BERT
文章目录 前言一、BERT1. Decoder-only2. Encoder-only3. Use of Bidirectional Context4. Masked Language Model (MLM)5. Next Sentence Prediction (NSP)6. Fine-tune1、情感分析2、句对分析3、命名实体识别(NER) 7. BERT总结 总结 前言 NLP选手对这…...
【力扣:新动计划,编程入门 —— 题解 ③】
—— 25.1.26 231. 2 的幂 给你一个整数 n,请你判断该整数是否是 2 的幂次方。如果是,返回 true ;否则,返回 false 。 如果存在一个整数 x 使得 n 2x ,则认为 n 是 2 的幂次方。 示例 1: 输入:…...
数据库基础:从概念到 MySQL 实战
数据库基础:从概念到 MySQL 实战 在当今数字化时代,数据的重要性不言而喻,而数据库作为数据管理的核心工具,发挥着关键作用。对于技术爱好者和开发者来说,深入了解数据库知识是必不可少的。今天,就让我们一…...
Nacos统一配置管理
Nacos 统一配置管理 将一部分配置信息写到Nacos配置文件中,可以统一管理,并且这个文件支持热加载。 新建配置步骤:配置管理 -> 配置列表 -> 新建配置 Data ID(配置文件名id) 命名规范:服务名称-de…...
Centos7系统php8编译安装ImageMagick/Imagick扩展教程整理
Centos7系统php8编译安装ImageMagick/Imagick扩展教程整理 安装php8安装ImageMagick1、下载ImageMagick2、解压并安装3、查看是否安装成功 安装imagick扩展包 安装php8 点我安装php8 安装ImageMagick 1、下载ImageMagick wget https://www.imagemagick.org/download/ImageMa…...
android的gradle
Gradle User Manual gradle官网 这里有个gradlew很有用,因为这个可以在窗口中运行gradlew脚本 gradlew 和 gradlew.bat 都是 Gradle Wrapper(Gradle 包装器) 的一部分,它们的作用是让项目可以使用 Gradle 而无需提前在系统中…...
2025美赛MCM数学建模A题:《石头台阶的“记忆”:如何用数学揭开历史的足迹》(全网最全思路+模型)
✨个人主页欢迎您的访问 ✨期待您的三连 ✨ 《石头台阶的“记忆”:如何用数学揭开历史的足迹》 目录 《石头台阶的“记忆”:如何用数学揭开历史的足迹》 ✨摘要✨ ✨引言✨ 1. 引言的结构 2. 撰写步骤 (1)研究背景 &#…...
Python3 【高阶函数】项目实战:5 个学习案例
Python3 【高阶函数】项目实战:5 个学习案例 本文包含 5 个关于“高阶函数”的综合应用项目,每个项目都包含完整的程序代码、测试案例和执行结果。具体项目是: 成绩统计分析单词统计工具简易计算器工厂任务调度器数据管道处理 项目 1&#…...
ES6 类语法:JavaScript 的现代化面向对象编程
Hi,我是布兰妮甜 !ECMAScript 2015,通常被称为 ES6 或 ES2015,是 JavaScript 语言的一次重大更新。它引入了许多新特性,其中最引人注目的就是类(class)语法。尽管 JavaScript 一直以来都支持基于…...
