参考资料


• 维基百科https://en.wikipedia.org/wiki/Flyweight_pattern

• 百度百科http://baike.baidu.com/link?url=RIhzC8-C6HUnAX6DqI5IfjawN3yhOckKvj8wWLL9zL4OAqK_I4xpi55aAjTpQdm7mKYZj7fKGW-I4-qXcNsj-q

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模式的更多相关文章

  1. 常见设计模式解析和实现(C++)FlyWeight模式

    作用:运用共享技术有效地支持大量细粒度的对象 UML结构图: 解析: Flyweight模式在大量使用一些可以被共享的对象的时候使用.比如,在QQ聊天时很多时候你懒得回复又不得不回复,一般会用一些客套 ...

  2. Flyweight 模式

    如果一个应用程序使用了太多的对象, 就会造成很大的存储开销. 特别是对于大量轻量级 (细粒度)的对象,比如在文档编辑器的设计过程中,我们如果为每个字母创建一个对象的话,系统可能会因为大量的对象而造成存 ...

  3. Structual设计--Flyweight模式

    1.意图 运用共享技术有效地支持大量细粒度的对象. 2.别名 无 3.动机 有些应用程序得意于在其整个设计过程中採用对象技术,但简单化的实现代价极大.如我们在使用word的时候.假设设置正文字体为:t ...

  4. Java 实现享元(Flyweight)模式

    /** * 字母 * @author stone * */ public class Letter { private String name; public Letter(String name) ...

  5. Flyweight模式_Java中23种设计模式

    —————————— ASP.Net+Android+IOS开发..Net培训.期待与您交流! —————————— 享元模式: Flyweight模式的有效性很大程度上取决于如何使用它以及在何处使用 ...

  6. Java设计模式(5)共享模式/享元模式(Flyweight模式)

    Flyweight定义:避免大量拥有相同内容的小类的开销(如耗费内存),使大家共享一个类(元类). 为什么使用共享模式/享元模式 面向对象语言的原则就是一切都是对象,但是如果真正使用起来,有时对象数可 ...

  7. 设计模式之——flyweight模式

    flyweight模式,又叫做享元模式. 顾名思义,享元模式就是共享一个元素. 百度百科 解释为: 享元模式(英语:Flyweight Pattern)是一种软件设计模式.它使用共享物件,用来尽可能减 ...

  8. Flyweight模式(亨元模式)

    这应该算是最好理解的一个设计模式了吧·················· 面向对象语言的原则就是一切都是对象,但是如果真正使用起来,有时对象数可能显得很庞大,比如,字处理软件,如果以每个文字都作为一个 ...

  9. 设计模式(二十)Flyweight模式

    当使用new关键字生成类的实例时,需要给其分配足够的内存空间.当程序中需要大量对象时,如果都是用new关键字来分配内存,将会消耗大量内存空间.Flyweight模式就是尽量避免new出实例,而是通过尽 ...

随机推荐

  1. SQL on Hadoop 的真相(2)

    转自:http://blog.jobbole.com/87159/ 这是一组系列博客,目的是详尽介绍 SQL-on-Hadoop .该系列的第一篇会介绍一些存储引擎和在线事务处理(简称 OLTP )相 ...

  2. OSG 中 相交測试 模块 工作流程及原理

    主要涉及三个类: 1. osgUtil::PolytopeIntersector // 详细不同算法实现类 2. osgUtil::IntersectionVisitor //用来遍历节点树的每一个节 ...

  3. Struts2_day03--向值栈放数据

    向值栈放数据 1 向值栈放数据多种方式 第一种 获取值栈对象,调用值栈对象里面的 set 方法 第二种 获取值栈对象,调用值栈对象里面的  push方法 第三种 在action定义变量,生成变量的ge ...

  4. onSaveInstanceState

    我们已经分析过Activity的启动流程,从中也分析了Activity的生命周期.而其中有一个生命周期方法:onSaveInstanceState方法,今天我们主要讲解一下onSaveInstance ...

  5. TClientDataSet数据源设置

    TClientDataSet数据源设置   TClientDataSet数据源设置

  6. 【POJ3621】Sightseeing Cows 分数规划

    [POJ3621]Sightseeing Cows 题意:在给定的一个图上寻找一个环路,使得总欢乐值(经过的点权值之和)/ 总时间(经过的边权值之和)最大. 题解:显然是分数规划,二分答案ans,将每 ...

  7. json写入到excel表

    1. 拼接返回的json数据 // 拼接需要下载报表的HTML,并返回html;reportHtml(reporttData) { let html = `<html xmlns:o=" ...

  8. Python全栈day21(作业针对一个文件进行查询修改删除的操作练习)

    需求,有一个配置文件test.conf内容如下 backend www1 server 1 server 2 backend www2 server 3 server 4 add [{'backend ...

  9. org.hibernate.ObjectDeletedException: deleted object would be re-saved by cascade 解决方案 (网络转载)

    前提是配置了cascade=all,依然报这种错误,其实出现这个错误的大多数情况根本不是像网上的帖子所说的是什么级联删除的问题,而且hibernate session关于实体生命周期操作的原因,这里明 ...

  10. sevlet实现反盗链

    有时候为了网站的版权和安全问题,我们需要为我们的网站应用设置防盗链,这样可以保证我们网站的一些资源的安全性.防盗链的主要是通过获取http的请求头referer的信息来和我们的网站地址做对比,如果相同 ...