flyweight模式
参考资料
• 维基百科:https://en.wikipedia.org/wiki/Flyweight_pattern
Flyweight模式简介
GoF:Use sharing to support large numbers of fine-grained objects efficiently.
GoF:使用共享的方式来高效地支持大量细粒度对象的使用。
Wikipedia:In computer programming, flyweight is a software design pattern. A flyweight is an object that minimizes memory use by sharing as much data as possible with other similar objects; it is a way to use objects in large numbers when a simple repeated representation would use an unacceptable amount of memory. Often some parts of the object state can be shared, and it is common practice to hold them in external data structures and pass them to the flyweight objects temporarily when they are used.
Wikipedia:在计算机编程中,享元是一种软件设计模式。一个享元是一个对象,该对象通过和其他相似对象共享尽可能多的数据来减少内存的使用。当对象的重复使用将导致令人无法接受的内存使用量时,享元模式是一种解决大量使用对象问题的方法。通常情况下,对象的部分状态是可以共享的,通常我们将这部分数据存储在外部的数据结构中,当需要使用时,再将它们临时性地传递给享元。
百度百科:享元模式(FlyWeight),运用共享技术有效的支持大量细粒度的对象。
Flyweight模式详解
• 设计意图
运用共享技术来有效地支持大量细粒度的对象使用。
• 结构图

• 结构说明
▶ Flyweight
抽象享元角色,为具体享元角色定义了接口,通过接口享元可以接收外部状态。
▶ ConcreteFlyweight
具体享元角色,实现抽象享元角色定义的接口。如果存在内部状态,则为内部状态提供存储空间。一个具体享元角色必须是可以共享的,它存储的任何状态都必须内蕴,这就是说状态必须与具体享元角色对象的上下文无关。
▶ UnsharedConcreteFlyweight
复合享元角色。实际情况中,并不是所有的享元子类型都需要被共享。抽象享元角色接口只是使共享行为可行,但并不强制要进行共享。这种情况十分常见:在某些享元对象结构层次中,复合享元对象将具体享元对象作为孩子。
▶ FlyweightFactory
享元工厂角色,创建并管理享元对象实体。享元工厂将确保享元能够被正确地共享。当客户端请求一个享元,享元工厂对象将提供一个已有的享元实例,如果不存在,则创建一个新的享元实体。
▶ Client
客户端角色。维护享元对象的引用,计算或存储享元对象的外部状态。
• 享元共享示意图

Facade模式举例
• 例子一
这个例子出自于Wikipedia:https://en.wikipedia.org/wiki/Flyweight_pattern
#include <map>
#include <list>
#include <string>
#include <sstream>
#include <iomanip>
#include <iostream>
using namespace std; // Instances of CoffeeFlavour will be the Flyweights.
class CoffeeFlavour
{
private:
string m_strFlavourName; public:
CoffeeFlavour(const string &strFlavourName = "Unknown") : m_strFlavourName(strFlavourName) {} public:
const char *GetFlavourName() const
{
return m_strFlavourName.c_str();
}
}; typedef const CoffeeFlavour * const CPCCoffeeFlavour; // Menu acts as a FlyweightFactory and cache for CoffeeFlavour Flyweight objects
class Menu
{
typedef map<string, CPCCoffeeFlavour> FlavoursMap;
typedef map<string, CPCCoffeeFlavour>::iterator FlavoursMapIterator; public:
~Menu()
{
FlavoursMapIterator it = m_FlavoursMap.begin();
for (; m_FlavoursMap.end() != it; it++)
{
CPCCoffeeFlavour cpcCoffeeFlavour = it->second;
if (cpcCoffeeFlavour)
{
delete cpcCoffeeFlavour;
}
}
m_FlavoursMap.clear();
} private:
FlavoursMap m_FlavoursMap; public:
CPCCoffeeFlavour LookUp(const string &strFlavourName)
{
FlavoursMapIterator it = m_FlavoursMap.find(strFlavourName);
if (m_FlavoursMap.end() == it)
{
m_FlavoursMap.insert( pair<string, CPCCoffeeFlavour>(strFlavourName, ::new CoffeeFlavour(strFlavourName)) );
} return m_FlavoursMap[strFlavourName];
} int GetCoffeeFlavourCount()
{
return m_FlavoursMap.size();
}
}; class Order
{
private:
int m_nTableNumber;
CPCCoffeeFlavour m_cpcCoffeeFlavour; public:
Order(int nTableNumber, CPCCoffeeFlavour cpcCoffeeFlavour) :
m_nTableNumber(nTableNumber), m_cpcCoffeeFlavour(cpcCoffeeFlavour) {} public:
void Serve()
{
cout << left;
cout << "Serving "
<< setw() << m_cpcCoffeeFlavour->GetFlavourName()
<< " to table "
<< setw() << m_nTableNumber << endl;
}
}; class CoffeeShop
{
private:
Menu m_CoffeeMenu;
list<Order> m_CoffeeOrders; public:
void TakeOrder(const string &strFlavourName, int nTableNumber)
{
CPCCoffeeFlavour cpcFlavour = m_CoffeeMenu.LookUp(strFlavourName);
m_CoffeeOrders.push_back(Order(nTableNumber, cpcFlavour));
} void Service()
{
list<Order>::iterator it = m_CoffeeOrders.begin();
for (; m_CoffeeOrders.end() != it; it++)
{
it->Serve();
}
} string Report()
{
stringstream ss;
ss << "Total CoffeeFlavour Objects Made: " << m_CoffeeMenu.GetCoffeeFlavourCount();
return ss.str();
}
}; int main()
{
CoffeeShop shop; shop.TakeOrder("Cappuccino", );
shop.TakeOrder("Frappe", );
shop.TakeOrder("Espresso", );
shop.TakeOrder("Frappe", );
shop.TakeOrder("Cappuccino", );
shop.TakeOrder("Frappe", );
shop.TakeOrder("Espresso", );
shop.TakeOrder("Cappuccino", );
shop.TakeOrder("Espresso", );
shop.TakeOrder("Frappe", );
shop.TakeOrder("Cappuccino", );
shop.TakeOrder("Espresso", ); shop.Service();
cout << shop.Report() << endl; return ;
}
CoffeeFlavour作为享元共享结构图如下:

flyweight模式的更多相关文章
- 常见设计模式解析和实现(C++)FlyWeight模式
作用:运用共享技术有效地支持大量细粒度的对象 UML结构图: 解析: Flyweight模式在大量使用一些可以被共享的对象的时候使用.比如,在QQ聊天时很多时候你懒得回复又不得不回复,一般会用一些客套 ...
- Flyweight 模式
如果一个应用程序使用了太多的对象, 就会造成很大的存储开销. 特别是对于大量轻量级 (细粒度)的对象,比如在文档编辑器的设计过程中,我们如果为每个字母创建一个对象的话,系统可能会因为大量的对象而造成存 ...
- Structual设计--Flyweight模式
1.意图 运用共享技术有效地支持大量细粒度的对象. 2.别名 无 3.动机 有些应用程序得意于在其整个设计过程中採用对象技术,但简单化的实现代价极大.如我们在使用word的时候.假设设置正文字体为:t ...
- Java 实现享元(Flyweight)模式
/** * 字母 * @author stone * */ public class Letter { private String name; public Letter(String name) ...
- Flyweight模式_Java中23种设计模式
—————————— ASP.Net+Android+IOS开发..Net培训.期待与您交流! —————————— 享元模式: Flyweight模式的有效性很大程度上取决于如何使用它以及在何处使用 ...
- Java设计模式(5)共享模式/享元模式(Flyweight模式)
Flyweight定义:避免大量拥有相同内容的小类的开销(如耗费内存),使大家共享一个类(元类). 为什么使用共享模式/享元模式 面向对象语言的原则就是一切都是对象,但是如果真正使用起来,有时对象数可 ...
- 设计模式之——flyweight模式
flyweight模式,又叫做享元模式. 顾名思义,享元模式就是共享一个元素. 百度百科 解释为: 享元模式(英语:Flyweight Pattern)是一种软件设计模式.它使用共享物件,用来尽可能减 ...
- Flyweight模式(亨元模式)
这应该算是最好理解的一个设计模式了吧·················· 面向对象语言的原则就是一切都是对象,但是如果真正使用起来,有时对象数可能显得很庞大,比如,字处理软件,如果以每个文字都作为一个 ...
- 设计模式(二十)Flyweight模式
当使用new关键字生成类的实例时,需要给其分配足够的内存空间.当程序中需要大量对象时,如果都是用new关键字来分配内存,将会消耗大量内存空间.Flyweight模式就是尽量避免new出实例,而是通过尽 ...
随机推荐
- Linux 复制、移动覆盖文件不提示
# vi ~/.bashrc 如果你看到如下内容,以下命令都会用别名执行了,就是说自动加了 -i 参数 alias rm='rm -i'alias cp='cp -i'alias mv='mv - ...
- 【UVa】Jump(dp)
http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&p ...
- java泛型之泛型边界
http://blog.csdn.net/renwuqiangg/article/details/51296621
- leetcode 153: Majority Element
Given an array of size n, find the majority element. The majority element is the element that appear ...
- codevs 5963 [SDOI2017]树点染色
[题解]: #include<cstdio> #include<cstring> #include<iostream> using namespace std; ...
- 【BZOJ4551】[Tjoi2016&Heoi2016]树 并查集
[BZOJ4551][Tjoi2016&Heoi2016]树 Description 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为1),有以下两 ...
- angular 封装公共方法
angular封装公共方法到service中间件,节省开发时间 layer.service.ts openAlert(callback) {// 传递回调函数 const dialogRef = th ...
- eslint常规语法检
"no-alert": 0,//禁止使用alert confirm prompt "no-array-constructor": 2,//禁止使用数组构造器 & ...
- DPK750针式打印机驱动,750u.dll下载
http://pan.baidu.com/s/1dD1SFvV DPK750针式打印机 下载
- 160506、Spring mvc新手入门(11)-返回json 字符串的其他方式
Spring MVC返回 json字符串的方式有很多种方法,这里介绍最简单,也是最常使用的两种方式 一.使用 PrintWriter printWriter 直接输出字符串到返回结果中 不需 ...