享元模式

享元(Flyweight、Cache)模式属于结构型模式的一种。

享元模式通过将对象的内部状态外部状态分开,尽量共享内部状态来减少对象的创建。内部状态是对象可以共享的部分,而外部状态是对象特有的、依赖于环境的部分。

享元模式旨在有效共享对象,避免重复创建相同内容的对象,减少内存开销,让你能在有限的内存中载入更多对象。

享元类的状态只能由构造函数的参数进行一次性初始化,它不能对其他对象公开其设置器或公有成员变量。

享元模式只是一种优化。 

享元模式在Java标准库中有很多使用,比如Byte、Integer

享元模式是通过工厂方法创建对象,在工厂方法内部,很可能返回缓存的实例,而不是新创建实例,从而实现不可变实例的复用。

在实际应用中,享元模式主要应用于缓存,直接返回内存中缓存的数据。

当需要创建大量相似的对象时,且这些对象的内部状态有很多相同部分时,可以使用享元模式。

享元模式通常有以下组成部分:

  • Flyweight(享元类):抽象享元类,负责存储共享的内部状态,定义接口以允许客户端访问内部状态。
  • ConcreteFlyweight(具体享元类):具体的享元类,负责实现共享对象的操作,存储内部状态。
  • FlyweightFactory(享元工厂):享元工厂类,用来管理和维护享元对象池。它确保享元对象的共享,避免重复创建相同对象。
  • Client(客户端):使用享元对象的代码,负责传递外部状态,并将共享对象传递给其他对象进行操作。

假如开发一个文字编辑器,我们使用享元模式将相同字符的显示对象共享。

1、享元类

public interface Flyweight {
    void display(String extrinsicState); // 外部状态,依赖客户端的具体状态
}

2、具体享元类

public class ConcreteFlyweight implements Flyweight {
    private String intrinsicState; // 内部状态
    
    public ConcreteFlyweight(String intrinsicState) {
        this.intrinsicState = intrinsicState;
    }

    @Override
    public void display(String extrinsicState) {
        System.out.println("Displaying " + intrinsicState + " with external state " + extrinsicState);
    }
}

3、享元工厂

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

public class FlyweightFactory {
    private Map<String, Flyweight> flyweights = new HashMap<>();

    public Flyweight getFlyweight(String intrinsicState) {
        // 享元工厂决定是否复用已有享元或者创建一个新的对象
        if (!flyweights.containsKey(intrinsicState)) {
            flyweights.put(intrinsicState, new ConcreteFlyweight(intrinsicState));
        }
        return flyweights.get(intrinsicState);
    }
}

4、客户端

public class Client {
    public static void main(String[] args) {
        FlyweightFactory factory = new FlyweightFactory();
        
        Flyweight f1 = factory.getFlyweight("A");
        f1.display("First Instance");

        Flyweight f2 = factory.getFlyweight("A");
        f2.display("Second Instance");

        Flyweight f3 = factory.getFlyweight("B");
        f3.display("Third Instance");
    }
}

享元模式的优缺点

优点:

  • 节省内存:通过共享相同的对象,减少内存占用。
  • 提高性能:减少对象的创建,减少内存垃圾回收的频率,从而提高性能。
  • 降低资源消耗:共享对象减少了系统的内存消耗,特别是在系统需要创建大量相似对象时。

缺点:

  • 增加复杂度:享元模式的实现较为复杂,需要设计共享对象池。

享元模式的设计思想是尽量复用已创建的对象,常用于工厂方法内部的优化。

可能很多同学觉得享元模式与单例模式很像。的确,但是他们也有区别。单例模式是不允许创建新实例。享元模式是要求实例不变,所以把 "应该创建一个新实例" 的操作优化为 "直接返回一个缓存的实例"。

单例对象可以是可变的。享元对象是不可变的。

纸上得来终觉浅,绝知此事要躬行。-- 烟沙九洲


posted @ 2024-12-04 23:19  烟沙九洲  阅读(298)  评论(0编辑  收藏  举报