Java二十三种设计模式-享元模式(12/23)

news2025/1/11 12:56:34

享元模式:高效管理大量对象的设计模式

引言

在软件开发中,有时需要处理大量相似或重复的对象,这可能导致内存使用效率低下和性能问题。享元模式提供了一种解决方案,通过共享对象的共同部分来减少内存占用。

基础知识,java设计模式总体来说设计模式分为三大类:

(1)创建型模式,共5种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式。

(2)结构型模式,共7种:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式。

(3)行为型模式,共11种:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式。

第一部分:享元模式概述

1.1 定义与特点

享元模式的基本定义

享元模式(英语:Flyweight Pattern)是一种软件设计模式。它使用共享物件,用来尽可能减少内存使用量以及分享资讯给尽可能多的相似物件;它适合用于只是因重复而导致使用无法令人接受的大量内存的大量物件。通常物件中的部分状态是可以分享。常见做法是把它们放在外部数据结构,当需要使用时再将它们传递给享元。

享元模式是一种结构型设计模式,旨在通过共享来减少创建大量相似或相同对象时的内存消耗。这种模式通过共享对象的共有部分来实现对象的复用,从而降低系统资源的使用。

特点

  • 共享不变部分:享元模式的核心在于识别对象的共有特性,并在多个对象之间共享这些特性,而不是为每个对象独立创建。
  • 减少对象数量:通过共享对象,减少了系统中对象的总数量,降低了内存占用和垃圾回收的负担。
  • 细粒度对象:特别适用于大量细粒度对象的场景,例如字体渲染、图形用户界面中的图标等。

1.2 应用场景

何时适合使用享元模式

  • 大量相似对象:当系统中存在大量相似或相同的对象时,使用享元模式可以显著减少内存消耗。
  • 对象创建成本高:如果对象的创建过程复杂或成本高昂,享元模式可以通过共享来降低这些成本。
  • 对象状态外部化:当对象的状态可以被外部化,即对象的行为可以通过外部状态参数化,而不是依赖于对象内部状态时。

应用实例

  • 文本编辑器:在文本编辑器中,字符可以作为享元,因为文档中可能存在大量重复的字符。
  • 图形界面:在GUI设计中,按钮、图标等界面元素可能在多个地方重复使用,通过享元模式可以减少这些元素的内存占用。
  • 游戏开发:游戏中的敌人、树木、建筑等元素可能在多个场景中重复出现,享元模式可以有效地管理这些资源。

享元模式通过共享对象的共有部分来减少内存占用,提高资源使用效率。然而,这种模式也增加了系统的复杂性,需要仔细设计以确保对象的共享部分不会影响其独立性。在下一部分中,我们将详细介绍享元模式的组成部分和实现方式。

第二部分:享元模式的组成

2.1 角色定义

享元(Flyweight)

  • 定义:享元是实现共享的对象,它包含了可以被多个对象共享的内部状态。
  • 特点:享元对象通常很轻量,不包含任何唯一性的状态。

享元工厂(Flyweight Factory)

  • 定义:享元工厂负责创建和管理享元对象,确保享元对象可以被正确地共享和复用。
  • 特点:享元工厂维护一个享元对象池,用于存储和检索享元对象。

客户端(Client)

  • 定义:客户端是使用享元对象的代码,它通过享元工厂来请求所需的享元对象。
  • 特点:客户端不直接创建享元对象,而是通过享元工厂来获取。

2.2 职责分配

享元的职责

  • 维护内部状态:享元对象负责维护可以被共享的状态,如不变的数据或行为。
  • 接受外部状态:享元对象可以接受外部状态,这些状态通常在享元对象被使用时传递。

享元工厂的职责

  • 创建享元对象:享元工厂负责创建享元对象,确保不会创建重复的对象。
  • 管理对象池:享元工厂维护一个对象池,存储已经创建的享元对象,以便复用。
  • 分配享元对象:享元工厂负责根据客户端的请求分配享元对象,确保对象的正确共享。

客户端的职责

  • 请求享元对象:客户端通过享元工厂请求所需的享元对象。
  • 传递外部状态:客户端负责传递享元对象所需的外部状态,以便享元对象可以正确执行操作。
  • 使用享元对象:客户端使用从享元工厂获取的享元对象来执行所需的任务。

通过这种职责分配,享元模式能够有效地管理大量对象,减少内存消耗,并提高系统性能。在下一部分中,我们将通过Java代码示例来展示享元模式的具体实现。

第三部分:享元模式的实现

3.1 Java实现示例

以下是使用Java语言实现享元模式的代码示例。在这个例子中,我们将创建一个简单的文本编辑器,其中字符是享元对象。

// 享元接口
interface Flyweight {
    void print(String extrinsicState);
}

// 具体享元类
class ConcreteFlyweight implements Flyweight {
    private String intrinsicState;

    public ConcreteFlyweight(String intrinsicState) {
        this.intrinsicState = intrinsicState;
    }

    @Override
    public void print(String extrinsicState) {
        System.out.println("Character: " + intrinsicState + " with intrinsic state: " + this.intrinsicState);
    }
}

// 享元工厂
class FlyweightFactory {
    private HashMap<String, Flyweight> flyweights = new HashMap<>();

    public Flyweight getFlyweight(String key) {
        if (!flyweights.containsKey(key)) {
            flyweights.put(key, new ConcreteFlyweight(key));
        }
        return flyweights.get(key);
    }
}

// 客户端代码
public class Client {
    public static void main(String[] args) {
        FlyweightFactory factory = new FlyweightFactory();

        Flyweight characterA = factory.getFlyweight("A");
        characterA.print("Bold");

        Flyweight characterB = factory.getFlyweight("B");
        characterB.print("Italic");
    }
}

3.2 设计考虑

享元对象的共享性

  • 内部状态:享元对象的内部状态应该是共享的,并且不随环境改变而改变。
  • 外部状态:享元对象的外部状态应该由客户端提供,并且可以随环境变化。

享元工厂的设计

  • 对象池管理:享元工厂需要管理一个对象池,以存储和复用享元对象。
  • 线程安全:在多线程环境中,享元工厂需要确保线程安全。

享元对象的创建

  • 延迟初始化:享元对象的创建可以延迟到第一次使用时,以节省资源。
  • 对象复用:确保享元对象在不再使用时可以被回收并复用于其他客户端。

享元模式的灵活性

  • 扩展性:享元模式应允许新类型的享元对象容易地添加到系统中。
  • 兼容性:新添加的享元对象应与现有客户端代码兼容。

避免过度共享

  • 识别共享部分:正确识别哪些部分可以共享,哪些部分应该是唯一的。
  • 性能权衡:过度共享可能导致性能问题,如同步开销。

享元模式的适用性

  • 对象数量:评估系统中对象的数量,确定是否值得使用享元模式。
  • 对象大小:如果对象较小,享元模式可能不会带来显著的性能提升。

通过考虑这些设计问题,你可以更有效地实现享元模式,确保它为你的应用程序带来内存和性能上的好处。在下一部分中,我们将探讨享元模式的使用场景、优点与缺点。

第四部分:享元模式的使用场景

4.1 内存敏感的应用

在内存敏感的应用中,享元模式发挥着至关重要的作用,尤其是在资源受限的环境中。

讨论在内存敏感的应用中享元模式的应用:

  • 移动设备:在移动应用开发中,由于内存限制,享元模式可以用来减少内存消耗。
  • 嵌入式系统:在嵌入式系统中,内存和处理能力有限,使用享元模式可以提高效率。
  • 大型数据处理:在处理大规模数据集时,享元模式可以减少创建和销毁对象的开销。

应用实例:

  • 图像处理软件:在图像处理中,像素可以作为享元对象,通过共享减少内存使用。
  • 游戏开发:游戏中的敌人或道具,如果具有相似的行为和外观,可以共享实现。

4.2 大量相似对象的管理

当应用程序需要管理大量相似的对象时,享元模式可以显著提高内存使用效率和性能。

分析在需要管理大量相似对象时,享元模式的优势:

  • 减少内存占用:通过共享相似对象的共有状态,减少了内存占用。
  • 提高性能:减少了对象创建和销毁的时间,提高了程序性能。
  • 简化对象管理:享元工厂提供了一个集中的点来管理对象的创建和复用。

应用实例:

  • 文本编辑器:文本中的字符如果重复出现,可以作为享元对象来处理。
  • 图形界面:GUI组件如按钮、图标等,如果样式和行为相同,可以共享实现。

优势详解:

  • 对象池:享元模式通过对象池来管理对象,避免了频繁的创建和销毁。
  • 细粒度控制:享元模式允许对对象的细粒度控制,可以针对特定状态进行优化。
  • 灵活性:享元模式提供了在不增加额外内存负担的情况下扩展对象功能的能力。

通过这些使用场景的讨论,我们可以看到享元模式在处理大量相似对象时的优势。然而,也要注意享元模式可能会增加系统的复杂性,并且在某些情况下可能不是最佳解决方案。在下一部分中,我们将探讨享元模式的优点与缺点。

第五部分:享元模式的优点与缺点

5.1 优点

减少内存消耗

  • 共享对象:通过共享对象的内部状态,减少了内存中对象的总数量。
  • 对象复用:享元模式允许对象在不同地方被复用,避免了重复创建相同对象。

提高性能

  • 减少创建时间:由于减少了对象的创建,享元模式可以降低系统的响应时间。
  • 快速响应:对象的快速获取和释放可以提高应用程序的交互性能。

降低系统成本

  • 资源优化:在资源受限的环境中,享元模式可以显著降低系统运行成本。

提高代码的可维护性

  • 集中管理:享元工厂提供了对象的集中管理,简化了对象的创建和维护。

易于扩展

  • 添加新对象:新类型的享元对象可以很容易地添加到系统中,而不影响现有代码。

5.2 缺点

增加系统复杂性

  • 设计复杂性:享元模式增加了设计的复杂性,需要仔细规划内部状态和外部状态的分离。

过度共享的风险

  • 共享状态管理:如果共享状态管理不当,可能会导致数据不一致或竞态条件。

灵活性降低

  • 修改困难:一旦享元对象被广泛使用,对其进行修改可能会变得困难。

性能权衡

  • 同步开销:在多线程环境中,共享对象可能需要额外的同步机制,这可能会影响性能。

不适用于所有场景

  • 独特对象:如果对象之间的差异性很大,享元模式可能不是最佳选择。

难以识别享元对象

  • 内部状态识别:有时难以识别哪些部分是共享的内部状态,哪些是外部状态。

可能影响缓存

  • 缓存策略:享元模式可能会影响系统的缓存策略和内存访问模式。

享元模式是一种强大的设计模式,可以显著减少内存消耗并提高性能,特别是在处理大量相似对象的场景中。然而,它也需要谨慎使用,以避免增加系统的复杂性和维护难度。在实际应用中,根据具体需求和场景选择是否使用享元模式是非常重要的。在下一部分中,我们将比较享元模式与其他设计模式,并提供一些最佳实践和建议。

第六部分:享元模式与其他模式的比较

6.1 与单例模式的比较

单例模式

  • 定义:单例模式确保一个类只有一个实例,并提供一个全局访问点。
  • 使用场景:当需要严格控制资源数量,如配置管理器或全局缓存时。

享元模式

  • 定义:享元模式通过共享来高效地支持大量细粒度对象的复用。
  • 使用场景:当需要创建大量相似或可共享的对象时。

对比

  • 对象数量:单例模式限制对象只能有一个实例,而享元模式可以有多个实例,但通过共享来减少数量。
  • 共享机制:单例模式不涉及对象的共享,享元模式则依赖于对象的共享。
  • 目的:单例模式用于控制对象的创建数量,享元模式用于减少对象的内存占用。

6.2 与原型模式的对比

原型模式

  • 定义:原型模式使用原型实例指定创建对象的种类,通过复制这些原型创建新的对象。
  • 使用场景:当创建新对象的成本较高,或者需要快速复制现有对象时。

享元模式

  • 定义:享元模式通过共享不变的内部状态来减少对象的创建。
  • 使用场景:适用于创建大量相似对象,且对象的内部状态可以共享。

对比

  • 复用方式:原型模式通过复制原型来创建新对象,享元模式通过共享已有对象来复用。
  • 共享程度:原型模式通常用于完全独立的复制,享元模式则侧重于共享不变的状态。

第七部分:享元模式的最佳实践和建议

7.1 最佳实践

合理划分内部状态和外部状态

  • 内部状态:应为共享状态,不随对象的使用环境改变而改变。
  • 外部状态:应为瞬时状态,可以随享元对象的使用而变化,不影响其他对象。

确保享元对象的不变性

  • 不变对象:保持享元对象的状态不可变,确保共享的安全性。

使用享元工厂管理对象

  • 对象池:享元工厂应维护一个对象池,以便存储和复用享元对象。

避免外部状态的误用

  • 参数化:确保外部状态通过参数传递,避免错误地修改享元对象的内部状态。

保持享元对象的轻量级

  • 轻量设计:享元对象应设计得尽可能轻量,避免包含大量资源消耗。

考虑线程安全

  • 并发访问:在多线程环境中使用享元模式时,确保享元对象的线程安全。

7.2 避免滥用

避免过度设计

  • 简单问题复杂化:对于不需要共享的对象,避免使用享元模式,以免增加不必要的复杂性。

避免共享不当

  • 错误共享:避免将不应该共享的状态共享,这可能会导致数据不一致。

避免忽视性能影响

  • 性能评估:在引入享元模式之前,评估其对性能的实际影响,确保它确实能带来性能上的提升。

7.3 替代方案

使用对象池

  • 对象复用:对象池是另一种管理对象生命周期的方式,适用于需要频繁创建和销毁对象的场景。

采用原型模式

  • 快速复制:当对象的创建成本较高时,可以使用原型模式来快速复制已有对象。

依赖注入

  • 控制反转:依赖注入可以减少对象创建的复杂性,提高系统的灵活性和可测试性。

状态模式

  • 状态变化:当对象的状态变化复杂时,可以使用状态模式来管理对象的状态转换。

享元模式与组合模式结合

  • 部分-整体层次:在需要表示部分-整体结构时,可以结合使用享元模式和组合模式。

享元模式是一种强大的设计模式,可以在需要高效管理大量对象时减少内存消耗和提高性能。然而,合理使用享元模式并避免其缺点是至关重要的。了解其替代方案可以帮助开发者根据具体需求和场景选择最合适的设计模式。在实际开发中,应根据具体情况灵活运用享元模式,以达到最佳的设计效果。

结语

享元模式是一种有效的设计模式,用于通过共享来减少大量对象的内存占用。通过本文的深入分析,希望读者能够对享元模式有更全面的理解,并在实际开发中做出合理的设计选择。

博主还写了其他Java设计模式文章,请各位大佬批评指正:

(一)创建型模式(5种):

Java二十三种设计模式-单例模式(1/23)

Java二十三种设计模式-工厂方法模式(2/23)

Java二十三种设计模式-抽象工厂模式(3/23)

Java二十三种设计模式-建造者模式(4/23)

Java二十三种设计模式-原型模式(5/23)

(二)结构型模式(7种): 

Java二十三种设计模式-适配器模式(6/23)

Java二十三种设计模式-装饰器模式(7/23)

Java二十三种设计模式-代理模式(8/23)

Java二十三种设计模式-外观模式(9/23)

Java二十三种设计模式-桥接模式(10/23)

Java二十三种设计模式-组合模式(11/23)

Java二十三种设计模式-享元模式(12/23)

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

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

相关文章

Apache OFBiz 曝出严重漏洞,允许预身份验证 RCE

近日&#xff0c;研究人员发现 Apache OFBiz 中存在一个新的关键漏洞&#xff0c;该漏洞是 Apache OFBiz 中的一个错误授权问题&#xff0c;被追踪为CVE-2024-38856。该漏洞影响 18.12.14 之前的版本&#xff0c;18.12.15 版本解决了该漏洞。 SonicWall 的安全研究员 Hasib Vh…

字节跳动发Seed-TTS语音合成模型,可模仿任意人的声音,效果逼真

前期我们介绍过很多语音合成的模型&#xff0c;比如ChatTTS&#xff0c;微软语音合成大模型等&#xff0c;随着大模型的不断进步&#xff0c;其合成的声音基本跟真人没有多大的区别。本期介绍的是字节跳动自家发布的语音合成模型Seed-TTS。 Seed-TTS 推理包含四个功能模块&…

JavaScript中判断变量的类型

数据类型 在 JavaScript 中有 8 种基本的数据类型&#xff08;7 种原始类型和 1 种引用类型&#xff09;&#xff0c;它们分别是&#xff1a; 原始类型/基本类型&#xff1a; Number&#xff0c;BigInt&#xff0c;String&#xff0c;Boolean&#xff0c;null&#xff0c;unde…

C++开发基础之深入理解C++中的两种单例模式实现——线程安全与效率的权衡

引言&#xff1a; 单例模式是设计模式中的一种&#xff0c;它保证一个类仅有一个实例&#xff0c;并提供一个全局访问点。在C中&#xff0c;实现单例模式的方式多种多样&#xff0c;但随着多线程应用的普及&#xff0c;如何确保单例模式在多线程环境下的线程安全性成为了一个重…

深度学习--图像分割UNet介绍及代码分析

UNet介绍 参考UNet网络介绍整体架构UNet过程输入编码器&#xff08;下采样&#xff09;中间特征表示解码器&#xff08;上采样&#xff09;输出 代码详解unetUP和Unet关系上采样模块——unetUp用于图像分割的卷积神经网络&#xff08;CNN&#xff09;架构模块——Unet类的定义初…

使用 Manim 创建一个二维坐标平面【NumberPlane】

NumberPlane 是 Manim 中用于创建一个二维坐标平面的类。它可以帮助用户在场景中可视化坐标轴、网格线以及其他数学概念。具体来说&#xff0c;它的功能包括&#xff1a; 坐标轴&#xff1a;NumberPlane 提供了 x 轴和 y 轴&#xff0c;通常是中心对称的&#xff0c;允许用户清…

深入探究Python反序列化漏洞:原理剖析与实战复现

在现代应用程序开发中&#xff0c;Python反序列化漏洞已成为一个备受关注的安全问题。反序列化是Python中用于将字节流转换回对象的过程&#xff0c;但如果没有妥善处理&#xff0c;攻击者可以通过精心构造的恶意数据&#xff0c;利用反序列化漏洞执行任意代码&#xff0c;进而…

前端day4-表单标签

<!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>day4-表单</title> </head> <body&g…

# 基于MongoDB实现商品管理系统(2)

基于MongoDB实现商品管理系统&#xff08;2&#xff09; 基于 mongodb 实现商品管理系统之准备工作 1、案例需求 这里使用的不是前端页面&#xff0c;而是控制台来完成的。 具体的需求如下所示&#xff1a; 运行 查询所有 通过id查询详情 添加 - 通过id删除 2、案例分析 程…

进程创建,进程消亡

虚拟地址&#xff1a;通过虚拟技术&#xff0c;将外部存储设备的一部分空间&#xff0c;划分给系统&#xff0c;作为在内存不足时临时用作数据缓存。当内存耗尽时&#xff0c;电脑就会自动调用硬盘来充当内存&#xff0c;以缓解内存的紧张。 练习: 编写一个代码实现,一个父…

OGG转MP3音频格式转换:6种免费音频转换器推荐

在如今的数字音乐时代&#xff0c;不同音频格式的兼容性问题常常让我们感到困扰。其中&#xff0c;OGG和MP3是两种常见的音频格式&#xff0c;但由于设备和平台的支持问题&#xff0c;我们经常需要将OGG转换为MP3格式。 本文将为您详细介绍OGG和MP3的区别&#xff0c;为什么需要…

Spring Boot集成protobuf快速入门Demo

1.什么是protobuf&#xff1f; Protobuf&#xff08;Protocol Buffers&#xff09;是由 Google 开发的一种轻量级、高效的数据交换格式&#xff0c;它被用于结构化数据的序列化、反序列化和传输。相比于 XML 和 JSON 等文本格式&#xff0c;Protobuf 具有更小的数据体积、更快…

数据结构:队列(含源码)

目录 一、队列的概念和结构 二、队列的实现 头文件 初始化 入队列和出队列 获取队头队尾元素 队列有效数据数及队列判空 队列的销毁 完整源码 dl.h dl.c 一、队列的概念和结构 队列是一种只允许在一端进行插入数据操作&#xff0c;在另一端进行删除数据操作的特殊线性…

重生之我 学习【数据结构之顺序表(SeqList)】

⭐⭐⭐ 新老博友们&#xff0c;感谢各位的阅读观看 期末考试&假期调整暂时的停更了两个多月 没有写博客为大家分享优质内容 还容各位博友多多的理解 美丽的八月重生之我归来 继续为大家分享内容 你我共同加油 一起努力 ⭐⭐⭐ 数据结构将以顺序表、链表、栈区、队列、二叉树…

多米诺和托米诺平铺

有两种形状的瓷砖&#xff1a;一种是2 x 1的多米诺形&#xff0c;另一种是形如L的托米诺形。两种形状都可以旋转。 给定整数 n &#xff0c;返回可以平铺 2 x n 的面板的方法的数量。返回对 10^9 7 取模 的值。 平铺指的是每个正方形都必须有瓷砖覆盖。两个平铺不同&#xff…

maven常用命令与常见问题汇总

文章目录 一、IDEA 下载依赖包源码报错Sources not found for:xxxx二、常用命令1、打包 一、IDEA 下载依赖包源码报错Sources not found for:xxxx 解决方案&#xff1a; 方案1、在 terminal 运行 mvn dependency:resolve -Dclassifiersources 命令 方案2、右键特定的pom文件…

论文概览 |《IJGIS》2024 Vol.38 issue4

本次给大家整理的是《International Journal of Geographical Information Science》杂志2024年第38卷第4期的论文的题目和摘要&#xff0c;一共包括8篇SCI论文&#xff01; 论文1 knowledge-constrained large language model interactable with GIS: enhancing public risk …

笔试题 day1

目录 快速io 统计2的个数 两个数组的交集 点击消除 快速io import java.util.*; import java.io.*;public class Main {public static PrintWriter out new PrintWriter(new BufferedWriter(new OutputStreamWriter(System.out)));public static Read in new Read();publ…

瑞_Linux防火墙相关命令_Windows远程连接虚拟机的服务失败_Linux防火墙端口开放

&#x1f64a; 前言&#xff1a;博主在学习使用虚拟机的过程中&#xff0c;常常碰到 Windows 远程连接虚拟机的服务失败的问题。比如想要在主机上连接虚拟机中的 MongoDB 服务的时候&#xff0c;服务器或者虚拟机一般都会默认开启防火墙&#xff0c;则会导致远程连接失败&#…

做一个能和你互动玩耍的智能机器人之七-接入对话和大模型

接入科大迅飞的语音识别&#xff1a; private void printResult(RecognizerResult results) {String text JsonParser2.parseIatResult(results.getResultString());String sn null;// 读取json结果中的sn字段try {JSONObject resultJson new JSONObject(results.getResult…