当前位置: 首页 > news >正文

篇十一:享元模式:共享细粒度对象

篇十一:“享元模式:共享细粒度对象”

设计模式是软件开发中的重要工具,享元模式(Flyweight Pattern)是结构型设计模式的一种。享元模式旨在通过共享细粒度的对象,减少内存消耗和提高性能。在设计模式学习中,享元模式是一个重要的概念。本文将介绍享元模式的原理和优点,并提供C++实现享元模式的示例代码。

开始本篇文章之前先推荐一个好用的学习工具,AIRIght,借助于AI助手工具,学习事半功倍。欢迎访问:http://airight.fun/。

另外有2本不错的关于设计模式的资料,分享出来与大家学习参考。
链接:https://pan.baidu.com/s/1RmhQF_o1CdK8U7s5KeILog?pwd=xc6d
提取码:xc6d

1. 享元模式的原理:
享元模式是一种结构型设计模式,其核心思想是通过共享细粒度的对象,减少内存消耗和提高性能。在软件开发中,有些对象可能会重复创建,造成内存资源的浪费。享元模式通过维护一个对象池来管理这些细粒度的对象,当需要创建对象时,先从对象池中查找是否已经存在该对象,如果存在则直接返回,如果不存在则创建新的对象并加入对象池中。

享元模式中有两种对象:内部状态和外部状态。内部状态是可以共享的,不会随着外部环境的改变而改变;而外部状态是随着外部环境的改变而改变的,每个对象都有不同的外部状态。

2. 享元模式的优点:

  • 减少内存消耗:通过共享细粒度的对象,减少了内存的消耗,提高了系统的性能。
  • 提高性能:共享对象减少了对象的创建和销毁过程,提高了系统的性能。
  • 支持大量细粒度对象:享元模式适用于大量细粒度对象的场景,将对象的内部状态共享,节省了内存空间。

3. 在C++中使用享元模式:

a. 定义享元类:

// Flyweight.h
#include <string>class Flyweight {
public:virtual ~Flyweight() {}virtual void operation(const std::string& extrinsicState) const = 0;
};

b. 创建具体享元类:

// ConcreteFlyweight.h
#include "Flyweight.h"
#include <iostream>class ConcreteFlyweight : public Flyweight {
public:void operation(const std::string& extrinsicState) const override {std::cout << "Concrete Flyweight with extrinsic state: " << extrinsicState << std::endl;}
};

c. 创建享元工厂类:

// FlyweightFactory.h
#include "Flyweight.h"
#include <unordered_map>class FlyweightFactory {
public:Flyweight* getFlyweight(const std::string& key) {if (flyweights_.find(key) == flyweights_.end()) {flyweights_[key] = new ConcreteFlyweight();}return flyweights_[key];}private:std::unordered_map<std::string, Flyweight*> flyweights_;
};

d. 使用享元模式:

// main.cpp
#include "Flyweight.h"
#include "ConcreteFlyweight.h"
#include "FlyweightFactory.h"int main() {FlyweightFactory factory;Flyweight* flyweight1 = factory.getFlyweight("shared");flyweight1->operation("state1");Flyweight* flyweight2 = factory.getFlyweight("shared");flyweight2->operation("state2");return 0;
}

在上述示例中,我们首先定义了享元类Flyweight,其中包含了操作的抽象接口operation()。然后,我们创建了具体享元类ConcreteFlyweight,实现了operation()接口,表示可以共享的细粒度对象。接着,我们创建了享元工厂类FlyweightFactory,用于管理享元对象的创建和共享。

main.cpp中,我们创建了享元工厂的实例factory,通过调用factory.getFlyweight()方法来获取享元对象。当需要创建对象时,首先查找是否已经存在相应的享元对象,如果存在则直接返回,如果不存在则创建新的对象。最后,我们调用flyweight1->operation()flyweight2->operation()来演示享元对象的使用。

4. 享元模式的代码解析:

  • 享元模式通过共享细粒度的对象,减少内存消耗和提高性能。
  • 共享对象分为内部状态和外部状态,内部状态是可以共享的,不随外部环境变化而变化;外部状态是随外部环境变化而变化的,每个对象都有不同的外部状态。
  • 享元模式适用于大量细粒度对象的场景,通过维护一个对象池来管理共享对象。

5. 总结:
享元模式是一种有用的设计模式,通过共享细粒度的对象,减少内存消耗和提高性能。在C++中,我们可以通过定义享元类、具体享元类和享元工厂类来应用享元模式。通过享元模式,我们可以有效地支持大量细粒度对象,并节省内存资源。享元模式的应用场景很多,例如在图形编辑器、文本编辑器或操作系统中都可以使用享元模式来优化性能。

希望本文能够帮助您深入理解享元模式的原理和优点,并通过C++的示例代码演示了如何实现享元模式。在后续的专栏文章中,我们将继续介绍更多设计模式的知识,包括原理、详细介绍、示例代码和代码解析,帮助您深入学习和应用设计模式。

参考文献:

  • Gamma, E., Helm, R., Johnson, R., & Vlissides,J. (1994). Design Patterns: Elements of Reusable Object-Oriented Software. Addison-Wesley Professional.

    • C++ Core Guidelines: https://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines

    6. 注意事项:
    在使用享元模式时,需要注意以下几点:

    • 区分内部状态和外部状态:在设计享元类时,需要区分哪些状态是内部状态可以共享的,哪些状态是外部状态随外部环境变化而变化的。确保在共享对象时,只共享内部状态,而外部状态是客户端负责传递的。
    • 线程安全性:如果享元对象在多线程环境下被并发访问,需要考虑线程安全性。可以使用锁机制或其他并发控制方法来保证共享对象的线程安全性。
    • 对象池管理:对象池的管理也需要考虑合适的策略,避免对象池过大导致内存浪费,或过小导致频繁的对象创建和销毁。

    7. 总结:
    享元模式是一种重要的设计模式,通过共享细粒度的对象,减少内存消耗和提高性能。在C++中,我们可以通过定义享元类、具体享元类和享元工厂类来应用享元模式。通过享元模式,我们可以支持大量细粒度对象,并节省内存资源。享元模式在图形编辑器、文本编辑器、操作系统等领域都有广泛的应用,能够优化系统性能,提升用户体验。

    希望本文能够帮助您深入理解享元模式的原理和优点,并通过C++的示例代码演示了如何实现享元模式。设计模式是软件开发中的重要知识,掌握不同的设计模式有助于提高代码质量、可维护性和可扩展性。在后续的专栏文章中,我们将继续介绍更多设计模式的知识,包括原理、详细介绍、示例代码和代码解析,帮助您深入学习和应用设计模式。

    如果您有其他需求或主题的建议,请随时告诉我,我将为您提供更多的帮助!

    参考文献:

    • Gamma, E., Helm, R., Johnson, R., & Vlissides, J. (1994). Design Patterns: Elements of Reusable Object-Oriented Software. Addison-Wesley Professional.
    • C++ Core Guidelines: https://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines

感谢您的阅读,欢迎一起探讨,共同进步,推荐大家使用学习助手AIRight来解答学习过程中的问题,访问链接:http://airight.fun/

http://www.lryc.cn/news/115502.html

相关文章:

  • Dev控件 Gridcontrol,gridview 实现多选功能
  • 内网穿透:如何通过公网访问本地Web服务器?
  • 在qemu中挂载镜像文件
  • 报错注入(主键重复)攻击原理
  • Golang基础教程
  • ppt压缩文件怎么压缩最小?文件压缩技巧分享
  • 实例033 制作闪烁的窗体
  • 【JavaEE进阶】Spring创建与使用
  • PHP8的循环控制语句-PHP8知识详解
  • 第八次作业
  • LeetCode //C - 290. Word Pattern
  • [保研/考研机试] 括号匹配问题 C++实现
  • springBoot集成caffeine,自定义缓存配置 CacheManager
  • 【瑞吉外卖】Git部分学习
  • 如何阐述自己做了一个什么样的东西
  • TC3XX - MCAL知识点(二十二):QSPI 同步与异步 Mcal配置及代码实战
  • led台灯哪些牌子性价比高?推荐几款性价比高的护眼台灯
  • 什么情况下容易发生锁表及如何处理
  • elk开启组件监控
  • Java Random 类的使用
  • 完美的分布式监控系统——Prometheus(普罗米修斯)与优雅的开源可视化平台——Grafana(格鲁夫娜)
  • pycharm的Terminal中如何设置打开anaconda3的虚拟环境
  • Jmeter(四) - 从入门到精通 - 创建网络测试计划(详解教程)
  • Flowable-结束事件-空结束事件
  • Elasticsearch:如何创建 Elasticsearch PEM 和/或 P12 证书?
  • 数仓架构模型设计参考
  • RedisTemplate.opsForGeo()用法简介并举例
  • Android OkHttp源码分析--拦截器
  • docker:如何传环境变量给entrypoint
  • kuboard安装和使用