「聊设计模式」之享元模式(Flyweight)

🏆本文收录于《聊设计模式》专栏,专门攻坚指数级提升,助你一臂之力,带你早日登顶🚀,欢迎持续关注&&收藏&&订阅!


前言

设计模式是解决复杂问题的最佳实践之一,这些模式都是由经验丰富的程序员在解决问题中总结出来的。享元模式是一种结构型设计模式,旨在通过共享尽可能多的对象来减少内存使用和提高性能。

摘要

享元模式是一个可用于优化重复性的设计模式。如果在应用程序中创建了大量相似的对象,那么可以使用享元模式来减少所需的内存和性能开销。该模式实现了对象的共享,从而减少了内存中对象的数量。也就是说,享元模式允许将对象的状态分为内部状态和外部状态。内部状态存储在享元对象内部,并且在对象创建之后不会发生变化。外部状态由客户端代码传递给享元对象。

享元模式

概念

享元模式是一种结构型设计模式,它通过共享相同或相似的对象来减少内存消耗,提高系统的性能和效率。

在享元模式中,对象分为两类:内部状态和外部状态。内部状态是对象的固有属性,可以被共享;而外部状态则取决于对象的使用环境,不可共享。享元模式将内部状态和外部状态分离,并共享内部状态,从而减少内存中相同状态对象的数量。

典型的享元模式应用场景是在需要大量创建相同或相似对象,且创建和销毁对象的成本较高时,如数据库连接池、线程池等。通过共享已有的对象,减少新对象的创建,可以有效地优化系统性能和资源利用。

结构

享元模式中有以下几个角色:

  • 抽象享元(Flyweight):享元接口,声明了享元对象的方法。
  • 具体享元(ConcreteFlyweight):实现抽象享元接口,实现抽象享元的方法,并且需要注意内部状态和外部状态的处理。
  • 享元工厂(FlyweightFactory):享元工厂用于创建和管理享元对象,它维护一个享元池用于存储已经创建的享元对象,以便重复利用。
  • 客户端(Client):使用享元工厂创建和管理享元对象。

如下是享元模式的UML类图:

优缺点

优点:

享元模式的优点如下:

  • 享元模式可以大大减少内存使用,因为共享对象可以由多个上下文使用。它可以用于缓存那些不经常改变但经常使用的对象,从而提高系统的性能。
  • 通过使用享元模式,可以减少对象的数量,从而降低系统的复杂性。

缺点:

享元模式的缺点如下:

  • 实现享元模式需要对系统进行全面分析和重构,这可能会增加系统的开发时间和成本。
  • 如果内部状态非常复杂,并且不能分为内部状态和外部状态,那么享元模式将无法提供帮助。

模式实现

下面是一个简单的示例代码,展示如何使用享元模式来创建和管理汽车对象。

首先是抽象享元接口:

java 复制代码
package com.example.javaDesignPattern.flyweight;

/**
 * @author bug菌
 * @version 1.0
 * @date 2023/9/19 16:15
 */
public interface Car {
    void drive(String location);
}

然后是具体享元实现:

java 复制代码
package com.example.javaDesignPattern.flyweight;

/**
 * @author bug菌
 * @version 1.0
 * @date 2023/9/19 16:15
 */
public class Bike implements Car {
    private String type;

    public Bike(String type) {
        this.type = type;
    }

    @Override
    public void drive(String location) {
        System.out.println("Drive " + type + " bike to " + location);
    }
}
java 复制代码
package com.example.javaDesignPattern.flyweight;

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

/**
 * @author bug菌
 * @version 1.0
 * @date 2023/9/19 16:16
 */
public class CarFactory {
    private static Map<String, Car> carPool = new HashMap<>();

    public static Car getCar(String type) {
        Car car = carPool.get(type);

        if (car == null) {
            car = new Bike(type);
            carPool.put(type, car);
            System.out.println("Create new car of type " + type);
        }
        return car;
    }
}

客户端代码如下所示:

java 复制代码
package com.example.javaDesignPattern.flyweight;

/**
 * @author bug菌
 * @version 1.0
 * @date 2023/9/19 16:16
 */
public class Client {
    public static void main(String[] args) {
        Car car1 = CarFactory.getCar("Sport");
        car1.drive("New York");

        Car car2 = CarFactory.getCar("Sport");
        car2.drive("San Francisco");

        Car car3 = CarFactory.getCar("Economy");
        car3.drive("Los Angeles");
    }
}

在客户端代码中,我们使用工厂方法来创建汽车对象。如果工厂已经创建了某种类型的汽车,则从享元池中返回汽车对象。否则,它将创建一个新的汽车对象,并将其添加到享元池中。

代码方法介绍

Car

  • drive(String location) 方法:驾驶汽车到指定的位置。

Bike

  • Bike(String type) 构造方法:创建具体享元对象。
  • drive(String location) 方法:驾驶自行车到指定的位置。

CarFactory

  • getCar(String type) 方法:从享元池中获取指定类型的汽车对象。

Client

  • main(String[] args) 方法:客户端代码,用于测试享元模式。

测试用例

下面是测试代码,用于测试享元模式是否工作正常。您可以在本地使用Java编译器运行该代码来查看它是否有效。

java 复制代码
public class Client {
    public static void main(String[] args) {
        Car car1 = CarFactory.getCar("Sport");
        car1.drive("New York");

        Car car2 = CarFactory.getCar("Sport");
        car2.drive("San Francisco");

        Car car3 = CarFactory.getCar("Economy");
        car3.drive("Los Angeles");
    }
}

执行结果如下:

小结

享元模式是一种结构性设计模式,可用于优化内存使用和性能。通过将对象分为内部状态和外部状态,可以共享已经创建的对象实例。该模式适用于需要创建大量相似对象的应用程序。

附录源码

如上涉及代码均已上传同步在GitHub,提供给同学们参考性学习。

总结

享元模式可以帮助我们优化重复性的代码。它允许我们共享已经创建的相似对象,并且可以显著减少内存使用和性能开销。但是,实现此模式需要对系统进行全面分析和重构。如果可以正确地实现享元模式,那么它将有助于提高系统的性能和扩展性。

☀️建议/推荐你

如果想系统性的全面学习设计模式,建议小伙伴们直接毫无顾忌的关注这个专栏《聊设计模式》,无论你是想提升自己的编程技术,还是渴望更好地理解代码背后的设计思想,本专栏都会为你提供实用的知识和启发,帮助你更好地解决日常开发中的挑战,将代码变得更加优雅、灵活和可维护!

📣关于我


我是bug菌,CSDN | 掘金 | infoQ | 51CTO 等社区博客专家,历届博客之星Top30,掘金年度人气作者Top40,51CTO年度博主Top12,华为云 | 阿里云| 腾讯云等社区优质创作者,全网粉丝合计15w+ ;硬核微信公众号「猿圈奇妙屋」,欢迎你的加入!免费白嫖最新BAT互联网公司面试题、4000G pdf电子书籍、简历模板等海量资料。

相关推荐
职教育人1 小时前
金砖软件测试赛项之Jmeter如何录制脚本!
java·测试工具·jmeter·性能优化·集成测试
码农小野3 小时前
基于SpringBoot的自习室预订系统
java·spring boot·后端
lizi888884 小时前
单组件的编写
java
java_heartLake4 小时前
设计模式之代理模式
java·设计模式·代理模式
ac-er88884 小时前
如何在Flask中实现国际化和本地化
后端·python·flask
Adolf_19934 小时前
Flask-WTF的使用
后端·python·flask
魏 无羡5 小时前
pgsql 分组查询方法
java·服务器·数据库
兩尛5 小时前
java--面向对象编程(中级部分)
java·开发语言
Xxxx. .Xxxx6 小时前
C语言程序设计实验与习题指导 (第4版 )课后题-第二章+第三章
java·c语言·开发语言
姜西西_6 小时前
[Spring]Spring MVC 请求和响应及用到的注解
java·spring·mvc