【设计模式 - 11】之享元模式(FlyWeight)
1、模式简介
当系统中存在大量对象时,非常容易造成内存溢出。为了解决这个问题,我们把这些对象中共有的部分抽象出来,如果有相同的业务请求,则直接返回在内存中已有的对象,避免重新创建,这就是享元模式。
享元模式(FlyweightPattern)主要用于减少创建对象的数量,以减少内存占用和提高性能,即它提供了减少对象数量从而改善应用所需的对象结构的方式。享元模式尝试重用现有的同类对象,如果未找到匹配的对象,则创建新对象。
例如,JAVA中的String使用的就是享元模式:
public class Test {
public static void main(String[] args) {
String a = "abc";
String b = "abc";
System.out.println(a == b);
}
}
// 返回值:true
又如,一个编辑器中如果只能输入大小写的字母,则有52个字符可以输入,此时我们只需要在编辑器中存储52个字符对象,而不需要每输入一个字符就创建一个对象。
享元模式的UML图:
享元模式的适用场景:
- 当系统中有大量对象时;
- 当这些对象基本相似时;
- 当这些对象会消耗大量内存时;
- 当这些对象需要使用缓冲池管理时。
享元模式的使用方法:
享元模式通常使用一个工厂管理,在工厂中维护一个HashTable或HashMap,使用唯一标识来存放和取出对象。另外还可以对工厂使用单例模式,保证项目中只有一个工厂。
享元模式的优点:
大大减少了对象的创建,降低了系统的内存,提高了效率。
享元模式的缺点:
提高了系统的负责度,需要分离出外部状态和内部状态,外部状态不应该随着内部状态的变化而变化,否则会造成系统的混乱。
2、实例
我们以上面说的编辑器的例子为需求,要求编辑器中只能输入字符,使用享元模式进行管理。
享元接口:
public interface MyChar {
String showMyChar();
}
享元实现类:
public class MyCharImpl implements MyChar {
private Character c; public MyCharImpl(Character c) {
this.c = c;
} @Override
public String showMyChar() {
return this.c + "";
}
}
享元工厂:
public class MyCharFactory {
private static MyCharFactory instance; private Map<Character, MyChar> charMap; private MyCharFactory() {
this.charMap = new HashMap<Character, MyChar>();
} /**
* 单例
*/
public static MyCharFactory getInstance() {
if (instance == null) {
synchronized (MyCharFactory.class) {
if (instance == null) {
instance = new MyCharFactory();
}
}
}
return instance;
} /**
* 根据字符的键获取字符对象
*/
public MyChar getMyChar(Character character) {
MyChar c = charMap.get(character);
if (c == null) {
c = new MyCharImpl(character);
charMap.put(character, c);
}
return c;
} /**
* 获取Map中存储的字符的数量
*/
public int getCharCount() {
return charMap.size();
}
}
测试类:
public class Test {
public static void main(String[] args) {
MyChar char1;
MyChar char2;
MyChar char3;
MyChar char4;
MyChar char5;
MyChar char6; char1 = MyCharFactory.getInstance().getMyChar(new Character('a'));
System.out.println("获取享元字符:" + char1.showMyChar());
System.out.println("当前享元字符数量:" + MyCharFactory.getInstance().getCharCount());
char2 = MyCharFactory.getInstance().getMyChar(new Character('b'));
System.out.println("获取享元字符:" + char2.showMyChar());
System.out.println("当前享元字符数量:" + MyCharFactory.getInstance().getCharCount());
char3 = MyCharFactory.getInstance().getMyChar(new Character('c'));
System.out.println("获取享元字符:" + char3.showMyChar());
System.out.println("当前享元字符数量:" + MyCharFactory.getInstance().getCharCount());
char4 = MyCharFactory.getInstance().getMyChar(new Character('a'));
System.out.println("获取享元字符:" + char4.showMyChar());
System.out.println("当前享元字符数量:" + MyCharFactory.getInstance().getCharCount());
char5 = MyCharFactory.getInstance().getMyChar(new Character('a'));
System.out.println("获取享元字符:" + char5.showMyChar());
System.out.println("当前享元字符数量:" + MyCharFactory.getInstance().getCharCount());
char6 = MyCharFactory.getInstance().getMyChar(new Character('d'));
System.out.println("获取享元字符:" + char6.showMyChar());
System.out.println("当前享元字符数量:" + MyCharFactory.getInstance().getCharCount());
}
}
运行结果如下图所示:
最后贴出享元模式的GitHub代码地址:【GitHub - FlyWeight】。
【设计模式 - 11】之享元模式(FlyWeight)的更多相关文章
- 设计模式系列之享元模式(Flyweight Pattern)——实现对象的复用
说明:设计模式系列文章是读刘伟所著<设计模式的艺术之道(软件开发人员内功修炼之道)>一书的阅读笔记.个人感觉这本书讲的不错,有兴趣推荐读一读.详细内容也可以看看此书作者的博客https:/ ...
- 设计模式学习心得<享元模式 Flyweight>
享元模式(Flyweight Pattern)主要用于减少创建对象的数量,以减少内存占用和提高性能.这种类型的设计模式属于结构型模式,它提供了减少对象数量从而改善应用所需的对象结构的方式. 享元模式尝 ...
- 《JAVA设计模式》之享元模式(Flyweight)
在阎宏博士的<JAVA与模式>一书中开头是这样描述享元(Flyweight)模式的: Flyweight在拳击比赛中指最轻量级,即“蝇量级”或“雨量级”,这里选择使用“享元模式”的意译,是 ...
- 设计模式-11享元模式(Flyweight Pattern)
1.模式动机 在面向对象程序设计过程中,有时会面临要创建大量相同或相似对象实例的问题.创建那么多的对象将会耗费很多的系统资源,它是系统性能提高的一个瓶颈. 享元模式就是把相同或相似对象的公共部分提取出 ...
- 第11章 享元模式(Flyweight Pattern)
原文 第11章 享元模式(Flyweight Pattern) 概述: 面向对象的思想很好地解决了抽象性的问题,一般也不会出现性能上的问题.但是在某些情况下,对象的数量可能会太多,从而导致了运行时 ...
- 设计模式(十)享元模式Flyweight(结构型)
设计模式(十)享元模式Flyweight(结构型) 说明: 相对于其它模式,Flyweight模式在PHP实现似乎没有太大的意义,因为PHP的生命周期就在一个请求,请求执行完了,php占用的资源都被释 ...
- 乐在其中设计模式(C#) - 享元模式(Flyweight Pattern)
原文:乐在其中设计模式(C#) - 享元模式(Flyweight Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 享元模式(Flyweight Pattern) 作者:weba ...
- 享元模式 FlyWeight 结构型 设计模式(十五)
享元模式(FlyWeight) “享”取“共享”之意,“元”取“单元”之意. 意图 运用共享技术,有效的支持大量细粒度的对象. 意图解析 面向对象的程序设计中,一切皆是对象,这也就意味着系统的运行将 ...
- 二十四种设计模式:享元模式(Flyweight Pattern)
享元模式(Flyweight Pattern) 介绍运用共享技术有效地支持大量细粒度的对象. 示例有一个Message实体类,某些对象对它的操作有Insert()和Get()方法,现在要运用共享技术支 ...
- 大话设计模式Python实现- 享元模式
享元模式(Flyweight Pattern):运用共享技术有效地支持大量细粒度的对象. 下面是一个享元模式的demo: #!/usr/bin/env python # -*- coding:utf- ...
随机推荐
- DevExpress 控件使用之XtraReport
DevExpress 系列控件,相信大家做WinForm开发已经再熟悉不过了.报表工具对大家来说,选择面很广,.net 本身也提供了非常好的设计工具.下面主要介绍通过DevExpress XtraRe ...
- oracle删除用户所有表
在删除数据表的时候往往遇到外键约束无法删除的情况,我们可以通过以下几步将数据库表删除,建议在删除库之前先对数据库进行备份,养成良好习惯. 1.删除外键 --查询用户所有表的外键,owner条件为use ...
- 谈memcache和memcached的区别
用了段时间的memcache和memcached总结下认识,看很多人在用cache的时候,刚刚都没有搞清楚memcache 和 memcached的区别,还有就是使用的时候基本都是 get/set ...
- php 执行linux 命令函数
php的内置函数exec,system都可以调用系统命令(shell命令),当然还有passthru,escapeshellcmd等函数. 在很多时候利用php的exec,system等函数调用系统命 ...
- #Leet Code# Unique Tree
语言:Python 描述:使用递归实现 class Solution: # @return an integer def numTrees(self, n): : elif n == : else: ...
- dictionary ----- python
Learn of dictionary,simple example of dictionary in “Simple Python tutorial"------------------ ...
- Swift(一,创建对象,类型推导,基本运算,逻辑,字符串,数组,字典)
swift用起来刚开始感觉有点怪怪的,但用了一段时间觉得还是挺好用的,哈哈.毕竟都是要有一个过程的嘛. 我就写一些自己在使用swift的时候的注意点吧,如有不正之处,还请指正! 一.在开发中优先使用常 ...
- Junit 源码剖析(一)
采用Junit4.8.2分析Junit实现架构 源码架构两个大包:junit包 org包 首先分析org.junit.runners.model包下的几个类 org.junit.runners.mod ...
- 从内部剖析C# 集合之---- HashTable
这是我在博客园的第一篇文章,写的不好或有错误的地方,望各位大牛指出,不甚感激. 计划写几篇文章专门介绍HashTable,Dictionary,HashSet,SortedList,List 等集合对 ...
- [BZOJ 3052] [wc2013] 糖果公园 【树上莫队】
题目链接:BZOJ - 3052 题目分析 这道题就是非常经典的树上莫队了,并且是带修改的莫队. 带修改的莫队:将询问按照 左端点所在的块编号为第一关键字,右端点所在的块为第二关键字,位于第几次修改之 ...