设计模式-享元模式

原文地址:LoveDev

享元模式(Flyweight Pattern):运用共享技术有效地支持大量细粒度对象的复用,系统只使用少量对象,而这些对象都很相似,状态变化很小,可以实现对象的多次复用。由于享元模式要求能够共享的对象必须是细粒度对象,因此它又称为轻量级模式,它是一种对象结构型模式。

当系统中存在大量相似或相同的对象时,将会导致运行代价过高、性能下降、OOM 等问题,享元模式正为解决之一类问题而诞生

在学习享元模式之前需要先了解一下 细粒度 和享元对象中的 内部状态外部状态 这三个概念:

  • 内部状态:不随环境改变而改变的状态,内部状态可以共享,例如人的性别,不管任何环境下都不会改变
  • 外部状态:随着环境改变而改变的状态,不可以共享的状态,享元对象的外部状态通常由客户端保存,并在享元对象创建后,需要的时候传入享元对象内部,不同的外部状态是相互独立的。例如衣服和鞋子,人在不同的环境下会穿不同的衣服和鞋子,但是衣服和鞋子又是相互独立不受彼此影响的
  • 细粒度:所包含的内部状态较少,逻辑较为简单,职责更加细分化
享元模式 UML
  • Flyweight(抽象享元类):通常是接口或抽象类,抽象享元类中声明了具体享元类公共方法,这些方法可以向外界提供享元对象的内部数据(内部状态),同时也可以通过这些方法来设置外部数据(外部状态)
  • ConcreteFlyweight(具体享元类):继承抽象享元类,在具体享元类中为内部状态提供存储空间。通常可以结合单例模式来设计具体享元类,为每一个具体享元类提供唯一的享元对象
  • UnshareConcreteFlyweight(非分享具体享元类):并不是所有的具体享元类都需要被共享,不能被共享的子类可以设计为非共享具体享元类,当需要一个非共享具体享元类的对象时可以直接通过实例化创建
  • FlyweightFactory(享元工厂类):创建并管理享元对象,将各种具体享元类存储到一个享元池中,享元池一般为“键值对”集合,可以结合工厂模式进行设计。当用户请求一个具体享元对象时,享元池中如果保存的有就直接返回给用户,如果没有就创建该享元对象返回给用户并存储到享元池中
俄罗斯方块

上面的图片是众所周知的俄罗斯方块中的一个个方块,这次就拿这个游戏举个栗子,如果在俄罗斯方块这个游戏中,每个不同的方块都是一个实例对象,这些对象就要占用很多的内存空间,下面利用享元模式进行改造:

Flyweight 类:

public abstract class AbstractBox {

    public abstract String getShape();

    public void display() {
        LogUtils.i("方块形状:" + this.getShape());
    }
}

ConcreteFlyweight 类:

// I形方块
public class IBox extends AbstractBox {
    @Override
    public String getShape() {
        return "I";
    }
}

// L形方块
public class LBox extends AbstractBox {
    @Override
    public String getShape() {
        return "L";
    }
}

// O形方块
public class OBox extends AbstractBox {
    @Override
    public String getShape() {
        return "O";
    }
}

FlyweightFactory 类:

public class BoxFactory {

    private static class SingletonHolder {
        private static final BoxFactory INSTANCE = new BoxFactory();
    }

    public static final BoxFactory getInstance() {
        return SingletonHolder.INSTANCE;
    }

    private static Hashtable<String, AbstractBox> sHashtable;

    private BoxFactory() {
        sHashtable = new Hashtable<>();

        AbstractBox iBox = new IBox();
        AbstractBox lBox = new LBox();
        AbstractBox oBox = new OBox();

        sHashtable.put("I", iBox);
        sHashtable.put("L", lBox);
        sHashtable.put("O", oBox);
    }

    public AbstractBox getBox(String key) {
        return sHashtable.get(key);
    }
}

Client 类:

AbstractBox i1 = BoxFactory.getInstance().getBox("I");
i1.display();
AbstractBox i2 = BoxFactory.getInstance().getBox("L");
i2.display();
AbstractBox i3 = BoxFactory.getInstance().getBox("O");
i3.display();
AbstractBox i4 = BoxFactory.getInstance().getBox("O");
i4.display();
// 用 == 对比两个对象的内存地址
LogUtils.i("两个对象是否相等:" + (i3 == i4));

可以看出,所有的方块都从工厂类中获取,而且是同一个对象,不用重新创建对象导致占用过多的内存。看完了之后内部状态的享元模式,下面接着看带有外部状态的享元模式,接下来给不同的方块染上不同的染色:

FlyweightFactory 类:

public class ExtrinsicStateBoxFactory {

    private static class SingletonHolder {
        private static final ExtrinsicStateBoxFactory INSTANCE = new ExtrinsicStateBoxFactory();
    }

    public static final ExtrinsicStateBoxFactory getInstance() {
        return SingletonHolder.INSTANCE;
    }

    private static Hashtable<String, ExtrinsicStateBox> sHashtable;

    private ExtrinsicStateBoxFactory() {
        sHashtable = new Hashtable<>();

        ExtrinsicStateBox jBox = new JBox();

        sHashtable.put("J", jBox);
    }

    public ExtrinsicStateBox getBox(String key) {
        return sHashtable.get(key);
    }
}

ConcreteFlyweight 类:

public class JBox extends ExtrinsicStateBox {
    @Override
    public String getShape() {
        return "J";
    }
}

Flyweight 类:

public abstract class ExtrinsicStateBox {
    public abstract String getShape();

    public void display(String color) {
        LogUtils.i("方块形状:" + this.getShape() + " 颜色:" + color);
    }
}

在调用 display() 的时候传入颜色的外部状态,虽然方块对象还是一个对象,但是它们可以具有不同的颜色

优点

  • 极大减少内存中相似或相同对象数量,节约系统资源,提供系统性能
  • 享元模式中的外部状态相对独立,且不影响内部状态

缺点

  • 为了使对象可以共享,需要将享元对象的部分状态外部化,分离内部状态和外部状态,使程序逻辑复杂

适用场景

  • 系统中具有大量相同或相似对象
  • 对象大部分状态都可以外部化
  • 享元池耗费一定系统资源,需要多次重复使用享元对象时才值得使用享元模式

源码地址:Github

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容

  • 定义 Flyweight在拳击比赛中指最轻量级,即“蝇量级”或“雨量级”。这里选择使用“享元模式”的意译,是因为这...
    步积阅读 1,610评论 0 2
  • 享元模式(Flyweight Pattern):运用共享技术有效地支持大量细粒度对象的复用。系统只使用少量的对象,...
    breezedancer阅读 1,913评论 2 51
  • 一、定义 享元模式(Flyweight Pattern)是池技术的重要实现方式,其定义如下:Use sharing...
    端木轩阅读 273评论 0 0
  • 享元模式——运用共享技术有效地支持大量细粒度的对象。享元模式可以避免大量相似类的开销,在软件开发中如果需要生成大量...
    语文小子阅读 472评论 0 0
  • 简介运用共享技术有效的支持大量细粒度的对象 应用:当一个应用使用了大量的对象,而大量的这些对象造成了很大的存储开销...
    Mitchell阅读 277评论 0 0