「全网最细 + 实战源码案例」设计模式——享元模式

ops/2025/2/3 18:44:08/

核心思想

  • 享元模式(Flyweight Pattern)是一种结构型设计模式,主要用于减少程序中大量对象的内存消耗。该模式通过共享相同的数据来有效减少内存的使用,适用于对象非常多且可以共享一部分状态的场景。
  • 核心:将对象的内部状态外部状态分离
    • 内部状态:存储在享元对象内部的、不会改变的状态,通常是可以共享的。
    • 外部状态:依赖于环境且变化的状态,不可共享。


结构

1. Flyweight(享元角色)

  • 声明享元对象的接口,通常是不可变的。

2. ConcretFlyweight(具体享元角色)

  • 实现 Flyweight 接口,存储共享状态。

3. UnsharedConcreteFlyweight(非共享具体享元角色)

  • 非共享的享元对象,一般不是由享元工厂创建。

4. FlyweightFactory(享元工厂)

  • 用来管理享元对象的工厂类,确保共享对象的唯一性。


适用场景

  1. 存在大量相似对象
  2. 对象状态可分为内部、外部状态
  3. 节省内存,避免重复创建对象。

优缺点

优点:

  1. 节省内存:避免重复创建相似对象。
  2. 提高性能:减少重复创建对象的开销。

缺点:

  1. 增加复杂性享元模式、工厂类的设计。

实现步骤

  1. 将需要改写为享元的类成员变量拆分为两个部分:
    1. 内在状态: 包含不变的、 可在许多对象中重复使用的数据的成员变量。
    2. 外在状态: 包含每个对象各自不同的情景数据的成员变量
  1. 保留类中表示内在状态的成员变量, 并将其属性设置为不可修改。 这些变量仅可在构造函数中获得初始数值。
  2. 找到所有使用外在状态成员变量的方法, 为在方法中所用的每个成员变量新建一个参数, 并使用该参数代替成员变量。
  3. 你可以有选择地创建工厂类来管理享元缓存池, 它负责在新建享元时检查已有的享元。 如果选择使用工厂, 客户端就只能通过工厂来请求享元, 它们需要将享元的内在状态作为参数传递给工厂。
  4. 客户端必须存储和计算外在状态 (情景) 的数值, 因为只有这样才能调用享元对象的方法。 为了使用方便, 外在状态和引用享元的成员变量可以移动到单独的情景类中。

示例

// 享元角色
public abstract class AbstractBox {// 获取图形public abstract String getShape();// 显示图形及颜色public void display(String color){System.out.println("方块:" + getShape() + ",颜色:" + color);}
}// 具体享元角色——I
public class IBox extends AbstractBox{@Overridepublic String getShape() {return "I";}
}// 具体享元角色——L
public class LBox extends AbstractBox{@Overridepublic String getShape() {return "L";}
}// 具体享元角色——O
public class OBox extends AbstractBox{@Overridepublic String getShape() {return "O";}
}// 享元工厂(静态内部类单例方式)
public class BoxFactory {// 享元对象池private Map<String, AbstractBox> boxes;// 私有化构造器private BoxFactory() {boxes = new HashMap<>();boxes.put("L", new LBox());boxes.put("I", new IBox());boxes.put("O", new OBox());}// 获取享元对象public AbstractBox getBox(String type) {return boxes.get(type);}// 创建静态内部类private static class SingletonHolder {private static final BoxFactory INSTANCE = new BoxFactory();}// 获取享元工厂实例public static BoxFactory getInstance() {return SingletonHolder.INSTANCE;}
}// 客户端
public class Client {public static void main(String[] args) {// 1.获取享元工厂BoxFactory boxFactory = BoxFactory.getInstance();// 2.获取享元对象AbstractBox box1 = boxFactory.getBox("L");AbstractBox box2 = boxFactory.getBox("I");AbstractBox box3 = boxFactory.getBox("O");AbstractBox box4 = boxFactory.getBox("O");// 3.获取非享元对象并显示box1.display("red");box2.display("blue");box3.display("green");box4.display("yellow");// 4.验证享元对象是否共享System.out.println(box3 == box4);}
}

在源码中的应用


与其他模式的关系

  • 你可以使用享元模式实现组合模式树的共享叶节点以节省内存。
  • 享元展示了如何生成大量的小型对象, 外观模式则展示了如何用一个对象来代表整个子系统。
  • 如果你能将对象的所有共享状态简化为一个享元对象, 那么享元就和单例模式类似了。 但这两个模式有两个根本性的不同。
    • 只会有一个单例实体, 但是享元类可以有多个实体, 各实体的内在状态也可以不同。
    • 单例对象可以是可变的。 享元对象是不可变的。

http://www.ppmy.cn/ops/155372.html

相关文章

小程序-基础加强-自定义组件

前言 这次讲自定义组件 1. 准备今天要用到的项目 2. 初步创建并使用自定义组件 这样就成功在home中引入了test组件 在json中引用了这个组件才能用这个组件 现在我们来实现全局引用组件 在app.json这样使用就可以了 3. 自定义组件的样式 发现页面里面的文本和组件里面的文…

STM32 旋转编码器

旋转编码器简介 旋转编码器&#xff1a;用来测量位置、速度或旋转方向的装置&#xff0c;当其旋转轴旋转时&#xff0c;其输出端可以输出与旋转速度和方向对应的方波信号&#xff0c;读取方波信号的频率和相位信息即可得知旋转轴的速度和方向 类型&#xff1a;机械触点式/霍尔传…

pandas(三)Series使用

一、Series基础使用 import pandasd {x:100,y:200,z:300} s1 pandas.Series(d) #将dict转化为Series print(s1)print("") l1 [1, 2, 3] l2 [a, b, c] s2 pandas.Series(l1, indexl2) #list转为Series print(s2)print("") s3 pandas.Series([11…

非根目录部署 nextjs 项目,资源文件 请求404 的问题

最近在学习next项目编写的代码放到服务器上静态资源404 先分析问题 到服务器上查看是有资源目录的是不是项目配置有问题是不是nginx配置有问题 经过排查1和2是没有问题的目前来看只有3 检查一下nginx配置 尝试着把静态资源的配置禁用 问题解决 我的next项目用的是pm2管理…

Josephus Problem II CSES - 2163

有3种方法 Solution 1 - ordered_set Utilizing the ordered_set This data structure is an extension of the general set in C. It allows searching for the K-th smallest element in O(log n) time complexity. #include <iostream> using namespace std; #…

【大数据技术】教程01:搭建完全分布式高可用大数据集群(VMware+CentOS+FinalShell)

搭建完全分布式高可用大数据集群&#xff08;VMwareCentOSFinalShell&#xff09; 资源下载 VMware Workstation Pro 16CentOS-Stream-10-latest-x86_64-dvd1.isoFinalShell 4.5.12 注&#xff1a;请在阅读本篇文章前&#xff0c;将以上资源下载下来。 写在前面 本章主要介…

C# 装箱和拆箱(以及 as ,is)

装箱&#xff08;Boxing&#xff09;是指将值类型转换为引用类型的过程 拆箱&#xff08;Unboxing&#xff09;是将引用类型转换回值类型的过程。 int a 1;object b a; //装箱object obj 10;int num (int)obj; //拆箱ArrayList list new ArrayList();list.Add(123);//装箱…

告别页面刷新!如何使用AJAX和FormData优化Web表单提交

系列文章目录 01-从零开始学 HTML&#xff1a;构建网页的基本框架与技巧 02-HTML常见文本标签解析&#xff1a;从基础到进阶的全面指南 03-HTML从入门到精通&#xff1a;链接与图像标签全解析 04-HTML 列表标签全解析&#xff1a;无序与有序列表的深度应用 05-HTML表格标签全面…