肝一肝设计模式【九】-- 享元模式

news2024/12/24 14:02:04

系列文章目录

肝一肝设计模式【一】-- 单例模式 传送门
肝一肝设计模式【二】-- 工厂模式 传送门
肝一肝设计模式【三】-- 原型模式 传送门
肝一肝设计模式【四】-- 建造者模式 传送门
肝一肝设计模式【五】-- 适配器模式 传送门
肝一肝设计模式【六】-- 装饰器模式 传送门
肝一肝设计模式【七】-- 代理模式 传送门
肝一肝设计模式【八】-- 外观模式 传送门


文章目录

  • 系列文章目录
  • 前言
  • 一、什么是享元模式
  • 二、享元模式中的角色
  • 三、举个栗子
  • 四、在开源框架中的使用
  • 写在最后


前言

本节我们继续分析设计模式中的结构型模式,前文中我们已经分析了适配器模式、装饰器模式、代理模式、外观模式,本节我们来学习一下——享元模式。


一、什么是享元模式

享元模式(Flyweight Pattern),它旨在减少对象的内存消耗,提高应用程序的性能。该模式通过共享尽可能多的细粒度对象来实现这一目标,从而有效地支持大量细粒度的对象。

在享元模式中,对象分为两种类型:内部状态(Intrinsic State)和外部状态(Extrinsic State)。内部状态是对象可以共享的状态,不依赖于对象的上下文,而外部状态是对象特定的、依赖于对象的上下文的状态。

享元模式的核心思想是将对象的状态划分为内部状态和外部状态,并且通过共享内部状态来减少对象的数量。

二、享元模式中的角色

享元模式包含以下几个角色:

  • 享元接口(Flyweight):定义享元对象的接口,通过该接口可以接收和操作外部状态。
  • 具体享元(ConcreteFlyweight):实现享元接口,包含内部状态,可以被共享和复用。具体享元对象需要注意线程安全的处理。
  • 享元工厂(FlyweightFactory):负责创建和管理享元对象,它维护一个享元池(Flyweight Pool),用于存储和复用已经创建的享元对象。当客户端请求一个享元对象时,享元工厂会检查享元池中是否已经存在该对象,如果存在则直接返回,否则创建一个新的享元对象并放入享元池中。
  • 客户端(Client):使用享元模式的对象,通过享元工厂获取享元对象,并根据需要传递外部状态。

三、举个栗子

最近在玩王国之泪,咱们就拿游戏来举个例子。ps:塞尔达是天,任天堂是宇宙的主宰!
在这里插入图片描述
在游戏中我们会遇到很多怪物,如果创建每个怪物都包含大量的数据(例如位置、外观、属性等),将会占用大量的内存。
而享元模式可以帮助我们减少内存消耗,提高性能。

上代码:

import java.util.HashMap;
import java.util.Map;

// 怪物接口
interface Monster {
    void display();
}

// 具体怪物类
class ConcreteMonster implements Monster {
    private String type;
    private String image; // 内部状态:怪物图片

    public ConcreteMonster(String type) {
        this.type = type;
        // 加载怪物图片,这里简化为字符串
        this.image = "Image_" + type;
    }

    public void display() {
        System.out.println("Type: " + type + ", Image: " + image);
    }
}

// 怪物工厂
class MonsterFactory {
    private static Map<String, Monster> monsterMap = new HashMap<>();

    public static Monster getMonster(String type) {
        if (monsterMap.containsKey(type)) {
            return monsterMap.get(type);
        } else {
            Monster monster = new ConcreteMonster(type);
            monsterMap.put(type, monster);
            return monster;
        }
    }
}

// 客户端代码
public class Client {
    public static void main(String[] args) {
        Monster monster1 = MonsterFactory.getMonster("波克布林");
        Monster monster2 = MonsterFactory.getMonster("丘丘");
        Monster monster3 = MonsterFactory.getMonster("波克布林");
        Monster monster4 = MonsterFactory.getMonster("丘丘");

        monster1.display(); // Type: 波克布林, Image: Image_波克布林
        monster2.display(); // Type: 丘丘, Image: Image_丘丘
        monster3.display(); // Type: 波克布林, Image: Image_波克布林
        monster4.display(); // Type: 丘丘, Image: Image_丘丘

        System.out.println("monster1 == monster3: " + (monster1 == monster3)); // true,内部状态相同的对象被共享
        System.out.println("monster2 == monster4: " + (monster2 == monster4)); // true,内部状态相同的对象被共享
    }
}

在上述示例中,我们定义了一个怪物接口 Monster,并实现了具体的怪物类 ConcreteMonster。ConcreteMonster 类包含了怪物的类型和内部状态 image,在构造函数中根据类型加载怪物图片。

然后,我们创建了怪物工厂 MonsterFactory,通过该工厂可以获取怪物对象。在工厂中,我们使用一个 monsterMap 来缓存已创建的怪物对象。如果请求的怪物对象已存在于缓存中,则直接返回;否则,创建一个新的怪物对象并存入缓存。

在客户端代码中,我们通过怪物工厂获取了不同类型的怪物对象,并调用 display() 方法展示怪物的信息。注意,相同类型的怪物对象被共享使用,因为它们具有相同的内部状态。

四、在开源框架中的使用

数据库连接池就是一个常见的应用享元模式的例子,其中HikariCP 是一个轻量级、高性能的数据库连接池框架,它在内部使用了享元模式来管理连接对象的复用。

在 HikariCP 的源码中,核心的享元模式应用是在 HikariPool 类中,它是连接池的主要实现。下面是一个简化的代码示例,展示了 HikariPool 中享元模式的部分实现:

public class HikariPool {
    // 内部状态:连接对象池
    private ConcurrentBag<PoolEntry> poolEntries;

    // ...

    public Connection getConnection() throws SQLException {
        // 从连接池中获取一个可用的连接
        PoolEntry entry = poolEntries.borrow(timeoutMs);
        return wrapConnection(entry);
    }

    private Connection wrapConnection(PoolEntry entry) {
        // 对连接对象进行包装,添加上下文信息
        Connection connection = entry.connection;
        // ...
        return connection;
    }

    // ...

    private static class PoolEntry {
        // 内部状态:数据库连接对象
        final Connection connection;

        // 外部状态:是否被占用
        volatile boolean isMarkedInUse;

        // ...

        PoolEntry(Connection connection) {
            this.connection = connection;
        }

        // ...
    }
}

在上述示例中,HikariPool 类表示连接池的主要实现。poolEntries 是连接对象池,其中的每个元素 PoolEntry 表示一个连接对象。PoolEntry 类中的 connection 属性表示数据库连接对象,而 isMarkedInUse 属性表示连接是否被占用。

当调用 getConnection() 方法时,HikariPool 会从连接池中获取一个可用的连接对象,并调用 wrapConnection() 方法对连接进行包装,添加上下文信息等处理。

在 PoolEntry 类中,connection 属性表示数据库连接对象,而 isMarkedInUse 属性用于标记连接是否被占用。这个外部状态的管理使得连接对象可以在不同的上下文中被复用。

由于源码非常复杂,上述示例只是简化了部分代码,真实的 HikariCP 源码中还包含了许多其他功能和优化策略。


写在最后

享元模式可以有效地节省内存空间,特别是在需要创建大量细粒度对象时。通过共享内部状态,可以减少对象的数量,提高应用程序的性能。然而,享元模式的使用需要权衡内部状态和外部状态的划分,以及对线程安全的处理,因为共享的对象可能被多个线程同时访问。

享元模式的优点:

  • 内存优化:享元模式通过共享对象的方式减少了内存的使用。相同的对象只在内存中存储一份,多个相似的对象可以共享相同的状态,从而减少了对象的数量和内存占用。
  • 性能提升:由于享元模式减少了对象的数量,从而减少了对象的创建和销毁的开销,提高了系统的性能。特别是在需要大量创建对象的场景下,享元模式可以显著减少系统的开销。
  • 状态外部化:享元模式将对象的内部状态和外部状态分离,内部状态共享,外部状态由客户端管理。这样可以简化对象的状态管理,提高系统的可维护性和灵活性。

享元模式的缺点:

  • 对象共享可能引发线程安全问题:由于多个线程共享相同的对象,可能需要额外的同步措施来保证线程安全性。在多线程环境下,需要注意对共享对象的访问控制,以避免并发问题。
  • 对象共享可能导致代码复杂性增加:享元模式需要对对象的内部状态和外部状态进行区分和管理,可能会增加代码的复杂性。这需要在设计和实现过程中进行权衡和折衷。

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

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

相关文章

java枚举类解读

目录 为什么需要枚举类 枚举类的使用 枚举类的实现 枚举类的属性 自定义枚举类 使用enum定义枚举类 常见enum的使用场景 switch 向枚举中添加新方法 覆盖枚举的方法 Enum类的主要方法 实现接口的枚举类 为什么需要枚举类 类的对象只有有限个&#xff0c;确定的。…

如何将map与对象进行转换

Spring Boot内置了一个强大的JSON转换器Jackson&#xff0c;可以实现将JSON字符串或Map类型的数据转换成Java对象。以下是将Map类型的数据转换成Java对象的示例代码&#xff1a; import com.fasterxml.jackson.databind.ObjectMapper;//定义Java对象 public class User {priva…

oracle客户端的安装教程

文章目录 一、安装前的准备工作 1.1、百度网盘安装包的连接 1.2、百度网盘oracle11g软件包 二、oracle数据库客户端的安装与数据的准备 安装步骤 前言 本文主要讲解oracle客户端的安装与简单使用过程 一、安装前的准备工作 1.1、百度网盘安装包的连接 客户端的软件包 …

STM32 FMC篇-SDRAM(IS42S16400J)

IS42S16400J 这个东西太常见啦&#xff0c;长方形的。不会过多解释&#xff0c;详细请阅读它的数据手册。 IS42S16400J是一种高速同步动态随机存储器(SDRAM)&#xff0c;64Mb的存储容量&#xff0c;采用4个bank&#xff0c;每个bank大小为16Mb&#xff0c;总线宽度为16位&…

【CW32开发】00 开发环境搭建和示例代码运行

1.下载MDK 2.下载芯片相关的固件库 我用的是芯片是CW32F030系列&#xff0c;所以下载相应的固件库 下载地址&#xff1a;https://www.whxy.com/support/filelist/13 3.安装固件库 解压下载的文件&#xff0c;并在cw32f030-stdperiph-lib\IdeSupport\MDK路径下安装固件库 …

064:cesium设置点划线材质(material-8)

第064个 点击查看专栏目录 本示例的目的是介绍如何在vue+cesium中设置点划线材质,请参考源代码,了解PolylineDashMaterialProperty的应用。 直接复制下面的 vue+cesium源代码,操作2分钟即可运行实现效果. 文章目录 示例效果配置方式示例源代码(共89行)相关API参考:专栏…

CBFS Vault 2022 for .NET Crack

将多个文件打包到一个 Vault - 一个“文件中的文件系统”&#xff0c;完成每个文件的压缩、透明加密和随机读/写访问。 亮点包括新的日记选项、用于更好地控制和跟踪的新事件&#xff0c;以及一系列核心性能和可用性改进 [了解更多]。 CBFS保险库 在任何地方存储一个完整的文件…

chatgpt赋能Python-pythone怎么下载

Python下载&#xff1a;在SEO中的作用 介绍 Python是一种广泛应用于编程、自动化任务和数据处理的强大语言。在SEO中&#xff0c;Python也扮演了越来越重要的角色。Python有许多库和工具可供下载&#xff0c;这使得它成为了SEO领域中必不可少的一部分。在本文中&#xff0c;我…

人工智能之读懂CNN卷积神经网络

通过往期文章的分享,我们了解了神经网络的结构,一般分为输入层,隐藏层,输出层 TensorFlow神经网络 那什么是卷积神经网络那,这就要我们追溯一下人类识别图像的原理 人类的视觉原理如下:从原始信号摄入开始(瞳孔摄入像素 Pixels),接着做初步处理(大脑皮层某些细胞发现…

chatgpt赋能Python-pythondna匹配

Python DNA匹配&#xff1a;从概念到应用 概述 DNA匹配是生物学中的一个重要领域&#xff0c;它意味着找到两个DNA序列之间的相似性并确定它们之间的关系。Python是一种被广泛用于生物信息学中的编程语言&#xff0c;它提供了一系列强大的库和工具来处理DNA序列数据&#xff…

【C++从0到王者】第七站:内存管理(520没有对象?那就new一个)

文章目录 一、C/C内存分区二、C语言中动态内存管理方式三、C内存管理方式1.new/delete操作内置类型2.C为什么要搞new和delete3.C中new和delete操作自定义类型4.非要乱用delete和free会造成什么后果&#xff1f; 四、 operator new与operator delete函数五、operator new和opera…

在安卓中压缩GIF的几种方法(附实例代码)

前言 最近在划水摸鱼的时候&#xff0c;看到有位大佬发了一篇 GIF 压缩思路的文章。 让我突然想起来&#xff0c;很久以前我在我的项目 隐云图解制作 中就实现了一个动图工具箱&#xff0c;其中一个功能就是压缩GIF。 不过这位大佬只介绍了其中几种使用方法&#xff0c;还有…

Java动态类型语言支持

JDK7发布字节码首位新成员——invokedynamic指令。以实现动态类型语言支持。也是为JDK8里可以顺利实现Lambda表达式而做的技术储备。我们将在本文详细了解动态语言支持这项特性出现的背景和它的意义与价值。 1 动态类型语言 动态类型语言的关键特征是它的类型检查的主体过程是…

MJ discord 添加应用配置

discord 添加机器人 https://discord.com/developers/applications 刷新token后显示&#xff0c;即机器人Token&#xff0c;后续配置到 mj.discord.bot-token 如图勾选后&#xff0c;打开url进行授权 选择Midjourney Bot所在的服务器 勾上这两个选项&#xff0c;点击 Save Cha…

图片转Excel表格,识别准确率的重要性

摘要&#xff1a;随着科技的不断发展&#xff0c;图片转Excel表格的应用越来越广泛。通过OCR技术实现图片转Excel表格&#xff0c;OCR识别准确率的提高对于信息录入的精度以及后续数据分析的可靠性具有非常重要的作用。本文探讨了影响OCR识别准确率的因素&#xff0c;并提出了提…

2.MATLAB篇——基本操作与矩阵输入

>> cos(((12345)^5)^0.5)ans -0.3623>> help sinsin - 参数的正弦&#xff0c;以弧度为单位此 MATLAB 函数 返回 X 的元素的正弦。sin 函数按元素处理数组。该函数同时接受实数和复数输入。 对于 X 的实数值&#xff0c;sin(X) 返回区间 [-1, 1] 内的实数值。 对于…

数据结构-单调栈2

这里是解了一个新的题目&#xff0c;然后对于有重复值的单调栈做了一些改进&#xff08;只适用于特殊题目&#xff09;&#xff1a;有重复值的单调栈不再使用ArrayList或者LinkedList&#xff0c;而是像无重复值的那样直接使用下标&#xff0c;这种方法能保证最终的正确性&…

Python图片转字符画,太好玩啦(46)

小朋友们好&#xff0c;大朋友们好&#xff01; 我是猫妹&#xff0c;一名爱上Python编程的小学生。 和猫妹学Python&#xff0c;一起趣味学编程。 今日主题 什么是字符画呢&#xff1f; 如何用Python将图片转为字符画&#xff1f; 我们先找一幅原画&#xff0c;比如它吧&…

亚马逊云科技 一周回顾 – 2022 年 7 月 18 日

上周&#xff0c;Amazon 峰会&#xff08;纽约&#xff09;在贾维茨中心线下举办&#xff0c;有数千名与会者以及 100 多家赞助商和合作伙伴参加。在主题演讲中&#xff0c;Amazon 首席开发人员倡导者 Martin Beeby 谈到了云基础设施的创新如何帮助客户适应挑战并抓住新的机遇。…

知识积累(1)

&#xff08;1&#xff09; 当您在Git中看到消息 "HEAD is now at 1343ccb FAB-17419 Fix off_chain_data sample error (#146)" 时&#xff0c;这是Git告知您当前所在的分支和最新的提交哈希。 这条消息通常出现在使用Git命令后&#xff0c;如git pull或git check…