掘金 后端 ( ) • 2024-04-17 09:51

theme: cyanosis

结构型模式

组成

Flyweight 模式通过共享对象,来有效地降低系统的实现成本,主要是降低存储成本、对象的创建成本。

UML 和组成 如下:

Flyweight: 享元对象的抽象接口;

ConcreteFlyweight:具体的享元对象;

Factory: 用于创建 Flyweight 对象的工厂,因为Flyweight对象要能够复用,所以不能每次由用户创建对象,只能通过唯一的入口来获取(创建)Flyweight对象,这个入口就是 FlyweightFactory。

因为创建的对象都是Flyweight对象,每个对象之间又存在差异(状态不同),所以Flyweight对象要有两个状态:外部状态和内部状态。内部状态是Flyweight对象自身的状态,不会被改变;外部状态是在使用Flyweight对象时,作为方法参数由外部环境传给Flyweight对象的。

Flyweight对象 是对重复对象消除外部状态后的创建的一个抽象,因此可以抽象成一个接口,具体的Flyweight对象则是接口的实现。

应用场景

在一些系统应用中,会创建大量的细粒度的、相似的对象,这些对象有着相似的组成结构,消除对象之间的差异后,这些对象可以表示为有限个基本对象集合。

对于这样的场景,可以使用 Flyweight模式,创建有限个基本对象(Flyweight对象),每次创建对象时都是复用已有的基本对象。

示例代码

Java 包装类 Integer 就使用了 享元模式,通过缓存 [-127,128] 之间的数,避免每次新建 Integer 对象。

public static Integer valueOf(int i) {
    // i 在 -127 和 128 之间,使用已经存在的缓存数据
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}

优点和缺点

使用 Flyweight 模式时,传输、查找和计算都会产生运行时的开销,节省的空间抵消了这些开销。

共享的 Flyweight 对象越多,节省的空间越大。

小结

Flyweight 通过重复使用已经存在的基本元素对象,减少重复创建对象的性能和空间损失。

Flyweight 对象和 枚举 的区别?

Flyweight对象和枚举很相似,都是用有限的集合来表示一些不会变的对象。但是,Flyweight对象和枚举是不一样的:

1、目的不一样

Flyweight对象是实现对象复用,从而降低存储和创建对象的成本;

枚举是对有限“取值”的描述,没有额外的效果。

2、含义不同

Flyweight对象本质上是“有状态的对象”;

枚举是对一些固定值的描述,是没有“状态”的,一个枚举出现在任何一个地方,含义都是相同的,但 Flyweight 对象只有内部状态是相同的,外部状态是不一样的。