java设计模式八 享元

享元模式(Flyweight Pattern)是一种结构型设计模式,它通过共享技术有效地支持大量细粒度的对象。这种模式通过存储对象的外部状态在外部,而将不经常变化的内部状态(称为享元)存储在内部,以此来减少内存中对象的数量。享元模式适用于那些可以共享的对象,以此来减少系统创建对象的数量,从而提升效率和性能。

### 案例分析:文字处理软件中的字符渲染

假设我们正在开发一个文字处理软件,需要显示大量的字符。每个字符可能有不同的颜色、字体和大小等属性,但字符本身(如'a', 'b', 'c'等)却是有限的。如果不使用享元模式,我们会为每个带有独特属性组合的字符创建一个对象,这会导致内存中存在大量重复的字符对象。

#### 享元模式的应用步骤:

1. **定义享元接口**:
   定义一个接口或抽象类,声明一个或多个方法用于操作内部状态,通常还会有一个方法来获取外部状态。

```java
public interface Character {
    void display(char character, Color color, Font font);
}
```

2. **实现享元类**:
   实现享元接口,并存储内部状态(在这个案例中,内部状态为空,因为字符是固定的)。

```java
public class ConcreteCharacter implements Character {
    @Override
    public void display(char character, Color color, Font font) {
        System.out.printf("Rendering character '%c' with color %s and font %s%n", character, color, font);
    }
}
```

3. **创建享元工厂**:
   工厂负责创建和管理享元对象,确保相同的字符复用同一个对象。为了高效地查找享元,通常会使用如HashMap这样的数据结构来存储已经创建的享元对象。

```java
public class CharacterFactory {
    private Map<Character, ConcreteCharacter> pool = new HashMap<>();

    public Character getCharacter(char character) {
        ConcreteCharacter flyweight = pool.get(character);
        if (flyweight == null) {
            flyweight = new ConcreteCharacter();
            pool.put(character, flyweight);
        }
        return flyweight;
    }
}
```

4. **客户端代码**:
   客户端从享元工厂获取享元对象,并设置外部状态(颜色、字体)后调用其方法。

```java
public class TextEditor {
    private CharacterFactory factory;

    public TextEditor() {
        factory = new CharacterFactory();
    }

    public void renderText(String text, Color color, Font font) {
        for (char c : text.toCharArray()) {
            Character character = factory.getCharacter(c);
            character.display(c, color, font);
        }
    }
}

// 使用示例
TextEditor editor = new TextEditor();
editor.renderText("Hello, World!", Color.RED, Font.TIMES_ROMAN);
```

在这个例子中,尽管每个字符在屏幕上显示时可能有不同的颜色和字体,但字符本身是有限的。享元模式通过`CharacterFactory`确保了对于每个不同的字符仅创建一个`ConcreteCharacter`实例,并通过传递外部状态(颜色和字体)来展示字符的不同表现,从而极大地减少了内存占用。

 

继续深入,我们可以探讨享元模式的一些关键优势、应用场景及注意事项,以加深对这一设计模式的理解。

### 关键优势

1. **内存优化**:最显著的优势是大幅度减少内存中对象的数量,特别是当存在大量相似对象时。这在处理大数据量或图形密集型应用时尤为重要,能有效减少内存占用,提高系统性能。

2. **性能提升**:减少了对象的创建和销毁过程,降低了垃圾回收的压力,提升了系统的运行速度。

3. **易于维护和扩展**:通过集中管理享元对象,使得对内部状态的修改变得更加集中和容易控制。同时,新的享元类型可以轻松添加到系统中,不影响现有代码。

### 应用场景

- **图形处理**:如上述字符渲染例子,图像处理软件中的像素、图标库等。
- **大量相似对象**:游戏开发中的子弹、棋盘上的棋子等,这些对象除了位置等少数状态外,其他大部分属性都相同。
- **字符串处理**:文本编辑器中的字符或单词,特别是在实现复杂格式化时。
- **网络编程**:连接池的设计中,复用数据库连接或socket连接等资源。

### 注意事项

1. **区分内外部状态**:正确区分对象的内部状态(被享元对象管理,不随环境改变)和外部状态(由客户端控制,随环境改变)。错误地处理状态可能会导致逻辑错误或数据混乱。

2. **线程安全**:当享元对象在多线程环境下被共享时,需要考虑线程安全问题。可能需要在享元类中加入同步机制或确保外部状态的线程安全性。

3. **初始化成本**:虽然享元模式能减少运行时的内存消耗,但如果享元对象的创建成本很高,初始化大量享元可能会影响程序的启动时间。

4. **过度设计**:在对象数量不是特别庞大,或者内存不是主要瓶颈的情况下,过度使用享元模式可能会增加系统的复杂度,得不偿失。因此,是否采用享元模式需根据实际情况权衡。

通过合理运用享元模式,可以在保证程序高效、内存友好的同时,保持代码的清晰与可维护性,是面向对象设计中一项非常实用的技巧。

 

接下来,让我们进一步探索享元模式在实际开发中的一些高级应用技巧和设计考量,以及如何与其他设计模式结合使用,以达到更好的效果。

### 高级应用技巧

1. **使用复合享元**:当享元对象需要组合其他享元对象以形成更复杂的结构时,可以引入复合享元的概念。例如,在游戏开发中,一个复合角色可能包含武器、防具等多个独立的享元对象,通过组合这些享元来构建更丰富的实体。

2. **懒加载**:为了进一步优化内存使用,可以在享元工厂中实现享元对象的懒加载策略。即首次请求某个享元时才创建它,而不是一开始就创建所有可能用到的享元对象。

3. **缓存管理**:对于内存敏感的应用,可以实施更精细的缓存管理策略,比如设置缓存上限,当缓存超出限制时,根据某种策略(如最近最少使用LRU)移除部分享元对象。

### 与其他设计模式结合

1. **与工厂模式结合**:如之前提到的,享元工厂负责创建和管理享元对象,这是工厂模式的一个典型应用。结合工厂模式,可以更加灵活和高效地控制享元对象的生命周期和分配。

2. **与代理模式结合**:当需要为享元对象添加额外的操作或控制访问时,可以通过代理模式包装享元对象。代理可以处理外部状态的传递,以及任何与具体业务逻辑相关的操作,同时保持享元对象本身的纯粹性。

3. **与状态模式结合**:如果享元对象的行为依赖于其状态,且这些状态变化较为复杂,可以引入状态模式来管理这些状态变化。享元对象的内部状态通过状态对象表示,状态对象的变化不会影响享元对象的身份,从而维持享元的共享特性。

### 总结

享元模式是一个强大的设计工具,它通过共享技术减少系统中对象的数量,优化了内存使用并提高了性能。在设计和实现时,需要细心区分内外部状态,考虑并发访问的线程安全问题,以及选择合适的时间和场景应用该模式。通过与其他设计模式的结合使用,可以解决更为复杂的问题,构建出更加高效、灵活的系统。掌握并灵活运用享元模式,是提升软件设计质量的重要一步。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

论文阅读】 ICCV-2021-3D Local Convolutional Neural Networks for Gait Recognition

motivation :现有方法方法无法准确定位身体部位&#xff0c;不同的身体部位可以出现在同一个条纹(如手臂和躯干)&#xff0c;一个部分可以出现在不同帧(如手)的不同条纹上。其次&#xff0c;不同的身体部位具有不同的尺度&#xff0c;即使是不同帧中的同一部分也可以出现在不同…

【光速上手 Hydra 】一行代码自动跑多次实验,Hydra 中的 Multirun 参数如何使用?

Hydra 是一个开源的 Python 框架&#xff0c;简化了研究和其他复杂应用的开发。其关键特性是能够通过组合动态地创建一个分层次的配置&#xff0c;并通过配置文件和命令行进行覆盖。Hydra 的名称来源于其能够运行多个类似的作业 - 就像一个有多个头的九头蛇一样。 主要特性&am…

泽众财务RPA机器人常见五个应用场景

泽众RPA&#xff08;即机器人流程自动化&#xff0c;Robotic Process Automation, RPA&#xff09;解决方案是依托于各类先进信息技术手段的虚拟劳动力 &#xff08;数字劳动力&#xff09;&#xff0c;根据预先设定的程序操作指令对任务进行自动化处理&#xff0c;实现业务流程…

解锁机械之美:发动机设备拆解可视化揭秘

在现代工程技术的世界里&#xff0c;发动机作为机械设备的心脏&#xff0c;始终扮演着至关重要的角色。图扑的发动机设备拆解可视化技术&#xff0c;以其独特的视角和精确的细节&#xff0c;为我们开启了一扇了解复杂机械结构的新窗。通过高清晰度的三维图像和动画&#xff0c;…

在阿里云K8S容器中,部署websocket应用程序的总结

一、背景 有一个websocket应用程序&#xff0c;使用spring boot框架开发&#xff0c;http端口号是6005&#xff0c;提供的是websocket服务&#xff0c;所以它还监听一个8889端口的tcp协议。 现在要把它部署到阿里云的k8s容器里&#xff0c;本文着重描述service层的配置。 因…

鸿蒙开发接口Ability框架:【 (ServiceExtensionAbility)】

ServiceExtensionAbility ServiceExtensionAbility模块提供ServiceExtension服务扩展相关接口的能力。 说明&#xff1a; 本模块首批接口从API version 9开始支持。后续版本的新增接口&#xff0c;采用上角标单独标记接口的起始版本。 本模块接口仅可在Stage模型下使用。 导入…

【linux软件基础知识】-死锁问题

死锁问题 当两个或多个线程由于每个线程都在等待另一个线程持有的资源而无法继续时,就会发生死锁 如下图所示, 在线程 1 中,代码持有了 L1 上的锁,然后尝试获取 L2 上的锁。 在线程 2 中,代码持有了 L2 上的锁,然后尝试获取 L1 上的锁。 在这种情况下,线程 1 已获取 L…

减速机齿数速算

1.齿轮相关参数 1.1 模数 &#xff0c; 因为 齿数*齿距 Pi*直径 所以&#xff1a;直径/齿数 齿距/PI 模数 国标现行标准&#xff08;截止2024/5&#xff09;是&#xff1a; GB/ 1357-2008 / ISO 54-1996 模数有国标的一个序列标准&#xff1a; 1.2.轴径 轴径的国标是&a…

【测试报告】星光日册

⭐ 作者&#xff1a;Jwenen &#x1f331; 作者主页&#xff1a;Jwenen的个人主页 &#x1f496; 持续更文&#xff0c;关注博主少走弯路&#xff0c;谢谢大家支持 &#x1f496; 测试报告 1. 项目介绍2. 测试用例框架3. 自动化测试源码 1. 项目介绍 “星光日册”项目实现了用…

追踪攻击数据包中的真实IP地址:方法与技巧

在网络安全领域&#xff0c;追踪攻击数据包中的真实IP地址是一项至关重要的任务。通过确定攻击者的真实IP地址&#xff0c;可以有效地识别和阻止网络攻击行为&#xff0c;提高网络安全防御水平。IP数据云IP地址查询将介绍几种常用的方法和技巧&#xff0c;帮助安全人员有效追踪…

【Linux】CAN根据时钟频率、波特率计算采样点详解

1、采样点知识回顾 参考博客:【CAN】知识点:帧类型、数据帧结构、传输速率、位时间、采样点 CAN 采样点是指在一个数据位的传输周期内,接收器实际采样数据的时间点。这个时间点是以百分比来表示的,它决定了在数据位的传输周期中,何时读取数据位的值。 正确设置采样点对…

60*13薪,外包到新疆...去吗?

大家好&#xff0c;我是白露呀。 今天我在牛客上看到一篇帖子&#xff0c;一位网友说自己收到一个 offer &#xff0c;薪资很高&#xff1a;60k*13&#xff0c;大约一年有近80万。 但是有个要求是外包到新疆的乌鲁木齐&#xff0c;他拿不定主意&#xff0c;就在牛客上发了这个…

华为数据之道第三部分导读

目录 导读 第三部分 第7章 打造“数字孪生”的数据全量感知能力 “全量、无接触”的数据感知能力框架 数据感知能力的需求起源&#xff1a;数字孪生 数据感知能力架构 基于物理世界的“硬感知”能力 “硬感知”能力的分类 “硬感知”能力在华为的实践 基于数字世界的…

Options API:选项式 API改成Composition API:组合式 API的留言板

让我欢喜让我忧 改成Composition API:组合式 API的代码&#xff0c; <template><!-- start --><span class"span_checkbox">操作<input type"checkbox" v-model"showInput" value"操作" /></span><…

VueReal将在Display Week上推出microLED创新技术

公司展示将microLED从晶圆转移到背板的“改变游戏规则”的平台 在2024年显示周&#xff08;5月12日至16日在圣何塞举行&#xff09;上&#xff0c;VueReal将展示其MicroSolid打印平台&#xff0c;并展示其在推动微LED显示器和其他微型半导体器件在智能手机显示器和AR/VR解决方案…

探索鸿蒙开发:鸿蒙系统如何引领嵌入式技术革新

嵌入式技术已经成为现代社会不可或缺的一部分。而在这个领域&#xff0c;华为凭借其自主研发的鸿蒙操作系统&#xff0c;正悄然引领着一场技术革新的浪潮。本文将探讨鸿蒙开发的特点、优势以及其对嵌入式技术发展的深远影响。 鸿蒙操作系统的特点 鸿蒙&#xff0c;作为华为推…

贪心算法-----柠檬水找零

今日题目&#xff1a;leetcode860 题目链接&#xff1a;点击跳转题目 分析&#xff1a; 顾客只会给三种面值&#xff1a;5、10、20&#xff0c;先分类讨论 当收到5美元时&#xff1a;不用找零&#xff0c;面值5张数1当收到10美元时&#xff1a;找零5美元&#xff0c;面值5张数…

jmeter控制器讲解

1&#xff0c;随机顺序控制器和随机控制器的区别&#xff1a;随机顺序控制器下所有的接口都会执行&#xff0c;只是执行顺序是随机的&#xff0c;随机控制器下所有的接口中随机执行一个接口&#xff0c;其余接口不执行。

C++之大数运算

溪云初起日沉阁 山雨欲来风满楼 契子✨ 我们知道数据类型皆有范围&#xff0c;一旦超出了这个范围就会造成溢出问题 今天说说我们常见的数据类型范围&#xff1a; 我们平时写代码也会遇到数据类型范围溢出问题&#xff1a; 比如 ~ 我们之前写的学生管理系统在用 int类型 填写…

学习笔记:IEEE 1003.13-2003(POSIX实时与嵌入式规范)

一、文档 在线参考&#xff1a; IEEE 1003.13-2003 免费下载Draft 版本&#xff08;pdf&#xff09;&#xff1a;IEEE Std. 1003.13 二、概念 1、POSIX标准 可移植操作系统接口&#xff08;英语&#xff1a;Portable Operating System Interface&#xff0c;缩写为POSIX&a…
最新文章