Java设计模式:享元模式实现高效对象共享与内存优化(十一)
目录
- 一、引言
- 二、享元设计模式的概念
- 1. 对象状态的划分
- 2. 共享机制
- 三、享元设计模式的组成
- 四、享元设计模式的工作原理
- 五、享元模式的使用
- 六、享元设计模式的优点和适用场景
- 结语
[参见]:
Java设计模式:核心概述(一)
Java设计模式:单例模式之六种实现方式详解(二)
Java设计模式:工厂模式之简单工厂、工厂方法、抽象工厂(三)
Java设计模式:建造者模式之经典与流式的三种实现(四)
Java设计模式:适配器模式的三种形式(五)
Java设计模式:深入装饰器模式的三种写法(六)
Java设计模式:组合模式之透明与安全的两种实现(七)
Java设计模式:代理模式的静态和动态之分(八)
Java设计模式:外观模式之优雅门面(九)
Java设计模式:组合优于继承化身之桥接模式(十)
一、引言
设计模式是一种用于解决在特定上下文中经常出现的问题的优秀解决方案。它们为开发人员提供了一种通用的设计语言,有助于创建灵活且可维护的代码。享元设计模式(Flyweight Pattern)是结构型模式之一,它通过共享对象来减少系统中对象的数量,从而降低系统资源消耗,提高系统性能。
二、享元设计模式的概念
享元设计模式(Flyweight Pattern)是众多设计模式中的一种,它专注于解决由于创建大量相似对象而导致的系统资源消耗过高的问题。这一模式的核心思想是“共享”,即通过共享尽可能多的对象来减少系统中对象的总数,进而降低内存占用和提高系统性能。
1. 对象状态的划分
在享元模式中,一个关键的概念是将对象的状态划分为两部分:内在状态(Intrinsic State)和外在状态(Extrinsic State)。
-
内在状态:这是对象内部固有的、不会随环境改变而变化的属性。内在状态通常是可以被共享的,因为它不会因外在条件的不同而有所区别。例如,在一个字符渲染系统中,字符的字体、大小等属性就可以被视为内在状态,因为不论在什么上下文中,相同字体和大小的字符都可以被复用。
-
外在状态:这是对象在特定上下文中才有的、会随着环境改变而变化的属性。外在状态是不能被共享的,因为它依赖于对象所处的具体环境和上下文。继续以字符渲染为例,字符的位置、颜色等可能因用户交互或页面布局的不同而变化,因此这些属性被视为外在状态。
2. 共享机制
享元模式通过实现一个享元工厂(Flyweight Factory)来管理对象的创建和共享。享元工厂会维护一个已创建享元对象的缓存(通常是一个哈希表或其他快速查找数据结构),并根据请求的类型(通常由内在状态决定)来返回相应的对象实例。如果请求的类型在缓存中已存在,则直接返回该实例;如果不存在,则创建一个新的享元对象,将其添加到缓存中,并返回给请求者。
- 提升性能与资源利用率:
通过上述共享机制,享元模式能够显著减少系统中相似对象的数量,从而降低内存占用。同时,由于避免了不必要的对象创建和销毁操作,系统的整体性能也会得到提升。这在处理大量相似对象时尤为重要,如文本编辑器中的字符渲染、图形界面中的图形元素绘制等场景。
综上所述,享元设计模式是一种通过共享对象来优化系统资源消耗和提高性能的设计方案。它通过将对象状态划分为内在状态和外在状态,并利用享元工厂来管理对象的创建和共享,从而实现了对相似对象的高效处理。
三、享元设计模式的组成

-
Flyweight(享元):这是享元对象,它封装了可以被共享的状态,并且可以被高效地复用。
-
FlyweightFactory(享元工厂):这个工厂负责创建和管理享元对象。它通常会缓存已创建的享元对象,并在请求相同类型的享元时返回已有的实例。
-
Client(客户端):客户端代码使用享元工厂来获取享元对象,并在需要时将其外在状态传递给享元。
四、享元设计模式的工作原理
享元模式的核心思想是“共享”。当系统中需要创建大量相似对象时,享元模式通过以下步骤来优化对象的创建和使用:
-
提取共享状态:首先,识别出对象中可以共享的状态,并将其封装在享元类中。
-
分离变化状态:将不能共享或随上下文变化的状态从享元类中分离出来,作为外在状态。
-
创建享元工厂:实现一个享元工厂来管理享元对象的创建和缓存。当客户端请求一个新的享元时,工厂首先检查缓存中是否已经存在相同类型的享元。如果存在,则返回该实例;否则,创建一个新的享元实例并添加到缓存中。
-
客户端使用:客户端代码通过享元工厂获取享元对象,并根据需要设置其外在状态。享元对象在执行操作时,会结合其内在状态和传递进来的外在状态来完成任务。
五、享元模式的使用
享元模式的实现通常涉及到一个享元类(表示要共享的对象),一个享元工厂类(负责创建和管理享元实例),以及客户端代码。下面代码展示如何使用享元模式来减少具有相同属性的对象的数量。

首先,定义享元类 Circle,表示一个圆形对象,其内在状态是颜色,这里假设所有圆的大小相同,因此不需要作为内在状态。
public class Circle {private String color; // 内在状态:颜色public Circle(String color) {this.color = color;}public String getColor() {return color;}// 假设的绘制方法,这里仅打印信息public void draw() {System.out.println("绘制一个颜色为 " + color + " 的圆形");}
}
然后,定义享元工厂类 CircleFactory,它负责管理并缓存已经创建的 Circle 对象。
import java.util.HashMap;
import java.util.Map;public class CircleFactory {private Map<String, Circle> circleMap = new HashMap<>(); // 用于缓存Circle对象的Map// 根据颜色获取Circle对象,如果缓存中没有则创建一个public Circle getCircle(String color) {Circle circle = circleMap.get(color);if (circle == null) {circle = new Circle(color);circleMap.put(color, circle); // 将新创建的Circle对象加入缓存}return circle;}
}
最后,编写客户端代码来使用享元工厂创建和使用享元对象。
public class FlyweightPatternDemo {private static final CircleFactory circleFactory = new CircleFactory();public static void main(String[] args) {// 从享元工厂获取圆形对象,并绘制它们Circle circle1 = circleFactory.getCircle("红色");circle1.draw(); // 输出:绘制一个颜色为 红色的圆形Circle circle2 = circleFactory.getCircle("蓝色");circle2.draw(); // 输出:绘制一个颜色为 蓝色的圆形Circle circle3 = circleFactory.getCircle("红色");circle3.draw(); // 输出:绘制一个颜色为 红色的圆形(注意这里是复用的circle1)// 验证circle1和circle3是否是同一个对象System.out.println(circle1 == circle3); // 输出:true}
}
Circle 类是享元,它的内在状态是颜色。CircleFactory 是一个享元工厂,它使用一个 HashMap 来缓存已经创建的 Circle 对象。当客户端请求一个具有特定颜色的圆形时,享元工厂首先检查缓存中是否已经有一个具有该颜色的圆形。如果有,则返回该对象;否则,创建一个新的 Circle 对象并将其添加到缓存中。通过这种方式,具有相同颜色的圆形对象实例在系统中只会被创建一次,从而减少了内存占用。
六、享元设计模式的优点和适用场景
优点:
-
减少内存消耗:通过共享对象实例,减少了系统中对象的数量,从而降低了内存消耗。
-
提高性能:避免了频繁的对象创建和销毁操作,提高了系统的运行性能。
-
简化系统设计:享元模式有助于将对象的共享部分和变化部分分离开来,使系统设计更加清晰和灵活。
适用场景:
-
当系统中需要处理大量相似对象时,如字符串、图形对象等。
-
当对象的创建成本较高,且大量对象之间存在大量重复数据时。
-
当需要优化系统资源消耗和提高系统性能时。
结语
享元设计模式是一种高效处理大量相似对象的设计方案。它通过共享对象实例来减少资源消耗和提高性能,适用于多种场景。在使用享元模式时,需要仔细识别对象的共享状态和变化状态,并合理设计享元类和享元工厂。通过合理运用享元模式,可以构建出更加灵活、高效和可维护的软件系统。

相关文章:
Java设计模式:享元模式实现高效对象共享与内存优化(十一)
码到三十五 : 个人主页 目录 一、引言二、享元设计模式的概念1. 对象状态的划分2. 共享机制 三、享元设计模式的组成四、享元设计模式的工作原理五、享元模式的使用六、享元设计模式的优点和适用场景结语 [参见]: Java设计模式:核心概述&…...
景源畅信电商:抖音开店步骤是什么?
随着社交媒体的兴起,抖音已经成为一个不可忽视的电商平台。许多人都希望通过抖音开店来实现自己的创业梦想。那么,抖音开店的具体步骤是什么呢?接下来,我们将详细阐述这一问题。 一、明确回答问题抖音开店的步骤主要包括:注册账号…...
Notepad++不显示CRLF的方法
View -> Show Symbol -> 去掉勾选 Show All Characters...
前端开发工程师——AngularJS
一.表达式和语句 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-w…...
【AI算法岗面试八股面经【超全整理】——概率论】
AI算法岗面试八股面经【超全整理】 概率论信息论机器学习CVNLP 目录 1、古典概型、几何概型2、条件概率、全概率公式、贝叶斯公式3、先验概率、后验概率4、离散型随机变量的常见分布5、连续型随机变量的常见分别6、数学期望、方差7、协方差、相关系数8、独立、互斥、不相关9.大…...
vue3 使用vant
使用前提: vite创建的vue3项目 vanthttps://vant-ui.github.io/vant/#/zh-CN/home npm i vant 引入样式: main.js import vant/lib/index.css vant封装 import { showLoadingToast,closeToast,showDialog,showConfirmDialog } from vant;export func…...
网络请求客户端WebClient的使用
在 Spring 5 之前,如果我们想要调用其他系统提供的 HTTP 服务,通常可以使用 Spring 提供的 RestTemplate 来访问,不过由于 RestTemplate 是 Spring 3 中引入的同步阻塞式 HTTP 客户端,因此存在一定性能瓶颈。根据 Spring 官方文档…...
unity制作app(9)--拍照 相册 上传照片
1.传输照片(任何较大的数据)都需要扩展服务器的内存空间。 2.还需要base64编码 2.1客户端发送位置的编码 2.2服务器接收部分的代码...
【busybox记录】【shell指令】mkfifo
目录 内容来源: 【GUN】【mkfifo】指令介绍 【busybox】【mkfifo】指令介绍 【linux】【mkfifo】指令介绍 使用示例: 创建管道文件 - 创建的时候同时指定文件权限 常用组合指令: 指令不常用/组合用法还需继续挖掘: 内容来…...
使用Jmeter进行性能测试的基本操作方法
🔥 交流讨论:欢迎加入我们一起学习! 🔥 资源分享:耗时200小时精选的「软件测试」资料包 🔥 教程推荐:火遍全网的《软件测试》教程 📢欢迎点赞 👍 收藏 ⭐留言 …...
Linux学习笔记(epoll,IO多路复用)
Linux learning note 1、epoll的使用场景2、epoll的使用方法和内部原理2.1、创建epoll2.2、使用epoll监听和处理事件 3、示例 1、epoll的使用场景 epoll的英文全称是extend poll,顾名思义是poll的升级版。常见的IO复用技术有select,poll,epo…...
STM32定时器及输出PWM完成呼吸灯
文章目录 一、STM32定时器原理1、基本定时器2、通用定时器(1)时钟源(2)预分频器PSC(3)计数器CNT(4)自动装载寄存器ARR 3、高级定时器 二、PWM工作原理三、控制LED以2s的频率周期性地…...
海外仓管理系统费用解析:如何选择高性价比的海外仓系统
海外仓作为链接国内商家和海外市场的重要环节,其重要性自然是不言而喻的。 对于众多中小型海外仓来说,如何在保证服务质量的同时降低运营成本,就成了大家关注的焦点。今天我们就从海外仓管理系统的费用这个角度,来帮助大家分析一…...
深度学习之学习率调度器Scheduler介绍
调度器是深度学习训练过程中非常重要的一部分,它用于动态调整模型的学习率,从而提高训练效率和最终性能。 1. 为什么需要学习率调度器? 深度学习训练中,学习率是一个非常关键的超参数。合适的学习率可以确保模型快速收敛并获得良好的性能。 但是在训练过程中,最优的学习率会随…...
蓝桥杯-AB路线(详细原创)
问题描述: 有一个由 N M 个方格组成的迷宫,每个方格写有一个字母 A 或者 B。小蓝站在迷宫左上角的方格,目标是走到右下角的方格。他每一步可以移动到上下左右相邻的方格去。 由于特殊的原因,小蓝的路线必须先走 K 个 A 格子、再…...
计算机字符编码的发展
目录 背景 发展 第一阶段:ASCII编码 第二阶段:扩展ASCII编码 第三阶段:各国编码 第四阶段:Unicode编码 第五阶段:UTF系列编码方式 相关扩展 背景 在计算机诞生初期,所有的数据都是基于二进制数&am…...
Java(六)——抽象类与接口
文章目录 抽象类和接口抽象类抽象类的概念抽象类的语法抽象类的特性抽象类的意义 接口接口的概念接口的语法接口的特性接口的使用实现多个接口接口与多态接口间的继承抽象类和接口的区别 抽象类和接口 抽象类 抽象类的概念 Java使用类实例化对象来描述现实生活中的实体&…...
【4.vi编辑器使用(下)】
一、vi编辑器的光标移动 二、vi编辑器查找命令 1、命令::/string 查找字符串 n:继续查找 N:反向继续查找 /^the 查找以the开头的行 /end 查找以 查找以 查找以结尾的行 三、vi编辑器替换命令 1、语法: : s[范围,范围]str1/str2[g] g表示全…...
【数据结构】探索树中的奇妙世界
专栏介绍: 哈喽大家好,我是野生的编程萌新,首先感谢大家的观看。数据结构的学习者大多有这样的想法:数据结构很重要,一定要学好,但数据结构比较抽象,有些算法理解起来很困难,学的很累…...
搭建YOLOv10环境 训练+推理+模型评估
文章目录 前言一、环境搭建必要环境1. 创建yolov10虚拟环境2. 下载pytorch (pytorch版本>1.8)3. 下载YOLOv10源码4. 安装所需要的依赖包 二、推理测试1. 将如下代码复制到ultralytics文件夹同级目录下并运行 即可得到推理结果2. 关键参数 三、训练及评估1. 数据结构介绍2. 配…...
使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
Linux链表操作全解析
Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表?1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...
React hook之useRef
React useRef 详解 useRef 是 React 提供的一个 Hook,用于在函数组件中创建可变的引用对象。它在 React 开发中有多种重要用途,下面我将全面详细地介绍它的特性和用法。 基本概念 1. 创建 ref const refContainer useRef(initialValue);initialValu…...
2.Vue编写一个app
1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...
Redis数据倾斜问题解决
Redis 数据倾斜问题解析与解决方案 什么是 Redis 数据倾斜 Redis 数据倾斜指的是在 Redis 集群中,部分节点存储的数据量或访问量远高于其他节点,导致这些节点负载过高,影响整体性能。 数据倾斜的主要表现 部分节点内存使用率远高于其他节…...
大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计
随着大语言模型(LLM)参数规模的增长,推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长,而KV缓存的内存消耗可能高达数十GB(例如Llama2-7B处理100K token时需50GB内存&a…...
从面试角度回答Android中ContentProvider启动原理
Android中ContentProvider原理的面试角度解析,分为已启动和未启动两种场景: 一、ContentProvider已启动的情况 1. 核心流程 触发条件:当其他组件(如Activity、Service)通过ContentR…...
MySQL 主从同步异常处理
阅读原文:https://www.xiaozaoshu.top/articles/mysql-m-s-update-pk MySQL 做双主,遇到的这个错误: Could not execute Update_rows event on table ... Error_code: 1032是 MySQL 主从复制时的经典错误之一,通常表示ÿ…...
文件上传漏洞防御全攻略
要全面防范文件上传漏洞,需构建多层防御体系,结合技术验证、存储隔离与权限控制: 🔒 一、基础防护层 前端校验(仅辅助) 通过JavaScript限制文件后缀名(白名单)和大小,提…...
Java并发编程实战 Day 11:并发设计模式
【Java并发编程实战 Day 11】并发设计模式 开篇 这是"Java并发编程实战"系列的第11天,今天我们聚焦于并发设计模式。并发设计模式是解决多线程环境下常见问题的经典解决方案,它们不仅提供了优雅的设计思路,还能显著提升系统的性能…...
