亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C++設計模式之享元模式

2020-05-23 14:20:41
字體:
來源:轉載
供稿:網友

前言

無聊的時候,也去QQ游戲大廳玩五子棋或者象棋;作為程序員,看到一個產品,總要去想想它是怎么設計的,怎么完成的,我想這個是所有程序員都會做的事情吧(強迫癥???)。有的時候,想完了,還要做一個DEMO出來,才能體現自己的NB,然后還有點小成就感。

在玩五子棋或象棋的時候,我就想過,騰訊那幫伙計是怎么做的呢?五子棋的棋子有黑白兩色,難道每次放一個棋子就new一個對象么?象棋有車、馬、相、士、帥、炮和兵,是不是每盤棋都要把所有的棋子都new出來呢?如果真的是每一個棋子都new一個,那么再加上那么多人玩;那要new多少對象啊,如果是這樣做的話,我想有多少服務器都是搞不定的,可能QQ游戲大廳會比12306還糟糕。那騰訊那幫伙計是如何實現的呢?那就要說到今天總結的享元模式了。

什么是享元模式?

在GOF的《設計模式:可復用面向對象軟件的基礎》一書中對享元模式是這樣說的:運用共享技術有效地支持大量細粒度的對象。

就如上面說的棋子,如果每個棋子都new一個對象,就會存在大量細粒度的棋子對象,這對服務器的內存空間是一種考驗,也是一種浪費。我們都知道,比如我在2013號房間和別人下五子棋,2014號房間也有人在下五子棋,并不會因為我在2013號房間,而別人在2014號房間,而導致我們的棋子是不一樣的。這就是說,2013號房間和2014號房間的棋子都是一樣的,所有的五子棋房間的棋子都是一樣的。唯一的不同是每個棋子在不同的房間的不同棋盤的不同位置上。所以,對于棋子來說,我們不用放一個棋子就new一個棋子對象,只需要在需要的時候,去請求獲得對應的棋子對象,如果沒有,就new一個棋子對象;如果有了,就直接返回棋子對象。這里以五子棋為例子,進行分析,當玩家在棋盤上放入第一個白色棋子時,此時由于沒有白色棋子,所以就new一個白色棋子;當另一個玩家放入第一個黑色棋子時,此時由于沒有黑色棋子,所以就需要new一個黑色棋子;當玩家再次放入一個白色棋子時,就去查詢是否有已經存在的白色棋子對象,由于第一次已經new了一個白色棋子對象,所以,現在不會再次new一個白色棋子對象,而是返回以前new的白色棋子對象;對于黑色棋子,亦是同理;獲得了棋子對象,我們只需要設置棋子的不同棋盤位置即可。

UML類圖

C++設計模式之享元模式

Flyweight:描述一個接口,通過這個接口flyweight可以接受并作用于外部狀態;

ConcreteFlyweight:實現Flyweight接口,并為定義了一些內部狀態,ConcreteFlyweight對象必須是可共享的;同時,它所存儲的狀態必須是內部的;即,它必須獨立于ConcreteFlyweight對象的場景;

UnsharedConcreteFlyweight:并非所有的Flyweight子類都需要被共享。Flyweight接口使共享成為可能,但它并不強制共享。

FlyweightFactory:創建并管理flyweight對象。它需要確保合理地共享flyweight;當用戶請求一個flyweight時,FlyweightFactory對象提供一個已創建的實例,如果請求的實例不存在的情況下,就新創建一個實例;

Client:維持一個對flyweight的引用;同時,它需要計算或存儲flyweight的外部狀態。

實現要點

根據我們的經驗,當要將一個對象進行共享時,就需要考慮到對象的狀態問題了;不同的客戶端獲得共享的對象之后,可能會修改共享對象的某些狀態;大家都修改了共享對象的狀態,那么就會出現對象狀態的紊亂。對于享元模式,在實現時一定要考慮到共享對象的狀態問題。那么享元模式是如何實現的呢?

在享元模式中,有兩個非常重要的概念:內部狀態和外部狀態。

內部狀態存儲于flyweight中,它包含了獨立于flyweight場景的信息,這些信息使得flyweight可以被共享。而外部狀態取決于flyweight場景,并根據場景而變化,因此不可共享。用戶對象負責在必要的時候將外部狀態傳遞給flyweight。

flyweight執行時所需的狀態必定是內部的或外部的。內部狀態存儲于ConcreteFlyweight對象之中;而外部對象則由Client對象存儲或計算。當用戶調用flyweight對象的操作時,將該狀態傳遞給它。同時,用戶不應該直接對ConcreteFlyweight類進行實例化,而只能從FlyweightFactory對象得到ConcreteFlyweight對象,這可以保證對它們適當地進行共享;由于共享一個實例,所以在創建這個實例時,就可以考慮使用單例模式來進行實現。

享元模式的工廠類維護了一個實例列表,這個列表中保存了所有的共享實例;當用戶從享元模式的工廠類請求共享對象時,首先查詢這個實例表,如果不存在對應實例,則創建一個;如果存在,則直接返回對應的實例。

代碼實現:

 

復制代碼代碼如下:

#include <iostream>
#include <map>
#include <vector>
using namespace std;
 
typedef struct pointTag
{
    int x;
    int y;
 
    pointTag(){}
    pointTag(int a, int b)
    {
        x = a;
        y = b;
    }
 
     bool operator <(const pointTag& other) const
     {
         if (x < other.x)
         {
             return true;
         }
         else if (x == other.x)
         {
             return y < other.y;
         }
 
         return false;
     }
}POINT;
 
typedef enum PieceColorTag
{
    BLACK,
    WHITE
}PIECECOLOR;
 
class CPiece
{
public:
    CPiece(PIECECOLOR color) : m_color(color){}
    PIECECOLOR GetColor() { return m_color; }
 
    // Set the external state
    void SetPoint(POINT point) { m_point = point; }
    POINT GetPoint() { return m_point; }
 
protected:
    // Internal state
    PIECECOLOR m_color;
 
    // external state
    POINT m_point;
};
 
class CGomoku : public CPiece
{
public:
    CGomoku(PIECECOLOR color) : CPiece(color){}
};
 
class CPieceFactory
{
public:
    CPiece *GetPiece(PIECECOLOR color)
    {
        CPiece *pPiece = NULL;
        if (m_vecPiece.empty())
        {
            pPiece = new CGomoku(color);
            m_vecPiece.push_back(pPiece);
        }
        else
        {
            bool bFound = false; // 非常感謝fireace指出的問題
            for (vector<CPiece *>::iterator it = m_vecPiece.begin(); it != m_vecPiece.end(); ++it)
            {
                if ((*it)->GetColor() == color)
                {
                    bFound = true;
                    pPiece = *it;
                    break;
                }
                bFound = false;
            }
            if (!bFound)
            {
                pPiece = new CGomoku(color);
                m_vecPiece.push_back(pPiece);
            }
        }
        return pPiece;
    }
 
    ~CPieceFactory()
    {
        for (vector<CPiece *>::iterator it = m_vecPiece.begin(); it != m_vecPiece.end(); ++it)
        {
            if (*it != NULL)
            {
                delete *it;
                *it = NULL;
            }
        }
    }
 
private:
    vector<CPiece *> m_vecPiece;
};
 
class CChessboard
{
public:
    void Draw(CPiece *piece)
    {
        if (piece->GetColor())
        {
            cout<<"Draw a White"<<" at ("<<piece->GetPoint().x<<","<<piece->GetPoint().y<<")"<<endl;
        }
        else
        {
            cout<<"Draw a Black"<<" at ("<<piece->GetPoint().x<<","<<piece->GetPoint().y<<")"<<endl;
        }
        m_mapPieces.insert(pair<POINT, CPiece *>(piece->GetPoint(), piece));
    }
 
    void ShowAllPieces()
    {
        for (map<POINT, CPiece *>::iterator it = m_mapPieces.begin(); it != m_mapPieces.end(); ++it)
        {
            if (it->second->GetColor())
            {
                cout<<"("<<it->first.x<<","<<it->first.y<<") has a White chese."<<endl;
            }
            else
            {
                cout<<"("<<it->first.x<<","<<it->first.y<<") has a Black chese."<<endl;
            }
        }
    }
 
private:
    map<POINT, CPiece *> m_mapPieces;
};
 
int main()
{
    CPieceFactory *pPieceFactory = new CPieceFactory();
    CChessboard *pCheseboard = new CChessboard();
 
    // The player1 get a white piece from the pieces bowl
    CPiece *pPiece = pPieceFactory->GetPiece(WHITE);
    pPiece->SetPoint(POINT(2, 3));
    pCheseboard->Draw(pPiece);
 
    // The player2 get a black piece from the pieces bowl
    pPiece = pPieceFactory->GetPiece(BLACK);
    pPiece->SetPoint(POINT(4, 5));
    pCheseboard->Draw(pPiece);
 
    // The player1 get a white piece from the pieces bowl
    pPiece = pPieceFactory->GetPiece(WHITE);
    pPiece->SetPoint(POINT(2, 4));
    pCheseboard->Draw(pPiece);
 
    // The player2 get a black piece from the pieces bowl
    pPiece = pPieceFactory->GetPiece(BLACK);
    pPiece->SetPoint(POINT(3, 5));
    pCheseboard->Draw(pPiece);
 
    /*......*/
 
    //Show all cheses
    cout<<"Show all cheses"<<endl;
    pCheseboard->ShowAllPieces();
 
    if (pCheseboard != NULL)
    {
        delete pCheseboard;
        pCheseboard = NULL;
    }
    if (pPieceFactory != NULL)
    {
        delete pPieceFactory;
        pPieceFactory = NULL;
    }
}

 

內部狀態包括棋子的顏色,外部狀態包括棋子在棋盤上的位置。最終,我們省去了多個實例對象存儲棋子顏色的空間,從而達到了空間的節約。

在上面的代碼中,我建立了一個CCheseboard用于表示棋盤,棋盤類中保存了放置的黑色棋子和白色棋子;這就相當于在外部保存了共享對象的外部狀態;對于棋盤對象,我們是不是又可以使用享元模式呢?再設計一個棋局類進行管理棋盤上的棋子布局,用來保存外部狀態。對于這個,這里不進行討論了。

優點

享元模式可以避免大量非常相似對象的開銷。在程序設計時,有時需要生成大量細粒度的類實例來表示數據。如果能發現這些實例數據除了幾個參數外基本都是相同的,使用享元模式就可以大幅度地減少對象的數量。

使用場合

Flyweight模式的有效性很大程度上取決于如何使用它以及在何處使用它。當以下條件滿足時,我們就可以使用享元模式了。

1.一個應用程序使用了大量的對象;
2.完全由于使用大量的對象,造成很大的存儲開銷;
3.對象的大多數狀態都可變為外部狀態;
4.如果刪除對象的外部狀態,那么可以用相對較少的共享對象取代很多組對象。

擴展

之前總結了組合模式組合模式,現在回過頭來看看,享元模式就好比在組合模式的基礎上加上了一個工廠類,進行共享控制。是的,組合模式有的時候會產生很多細粒度的對象,很多時候,我們會將享元模式和組合模式進行結合使用。

總結

使用享元模式可以避免大量相似對象的開銷,減小了空間消耗;而空間的消耗是由以下幾個因素決定的:

1.實例對象減少的數目;
2.對象內部狀態的數目;對象內部狀態越多,消耗的空間也會越少;
3.外部狀態是計算的還是存儲的;由于外部狀態可能需要存儲,如果外部狀態存儲起來,那么空間的節省就不會太多。

共享的Flyweight越多,存儲節約也就越多,節約量隨著共享狀態的增多而增大。當對象使用大量的內部及外部狀態,并且外部狀態是計算出來的而非存儲的時候,節約量將達到最大。所以,可以使用兩種方法來節約存儲:用共享減少內部狀態的消耗;用計算時間換取對外部狀態的存儲。

同時,在實現的時候,一定要控制好外部狀態與共享對象的對應關系,比如我在代碼實現部分,在CCheseboard類中使用了一個map進行彼此之間的映射,這個映射在實際開發中需要考慮的。

好了,享元模式就總結到這里了。希望大家和我分享你對設計模式的理解。我堅信:分享使我們更進步。
PS:至于騰訊那幫伙計到底是如何實現QQ游戲大廳的,我也不知道,這里也完全是猜測的,請不要以此為基準。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美亚洲一区在线| 亚洲福利小视频| 日韩欧美国产网站| 亚洲成人999| 中文字幕一区电影| 两个人的视频www国产精品| 欧美最猛性xxxxx免费| 欧美日韩一区二区免费视频| 欧美在线观看网站| 高清在线视频日韩欧美| 国产精品极品在线| 久久久久久久久久久亚洲| 日韩精品视频在线播放| 久久成年人视频| 最近2019免费中文字幕视频三| 亚洲伊人久久大香线蕉av| 国产成人精品优优av| 亚洲第一福利视频| 亚洲成av人片在线观看香蕉| 国内自拍欧美激情| 日韩影视在线观看| 在线午夜精品自拍| 亚洲欧美国产va在线影院| 成人有码在线视频| 国产精品网址在线| 日韩成人在线视频观看| 97在线看免费观看视频在线观看| 97在线免费观看视频| 国产精品美女久久久久av超清| 国产亚洲精品久久久久久牛牛| 国产精品亚洲综合天堂夜夜| 亚洲精品有码在线| 成人做爰www免费看视频网站| 57pao精品| 日韩av免费看| 国产亚洲a∨片在线观看| 欧美日韩福利电影| 色婷婷av一区二区三区在线观看| 国产精品青青在线观看爽香蕉| 欧美极品美女视频网站在线观看免费| 中日韩美女免费视频网址在线观看| 91精品久久久久久久久久久| 97久久精品在线| 日韩成人在线电影网| 国产精品v片在线观看不卡| 97视频国产在线| 国产成人午夜视频网址| 91麻豆国产精品| 国产亚洲欧洲高清| 亚洲国产精品人久久电影| 最好看的2019年中文视频| 色老头一区二区三区在线观看| xxxxx成人.com| 精品国产户外野外| 热久久免费国产视频| 亚洲综合最新在线| 日本一区二区不卡| 午夜精品一区二区三区在线视| 国产精品久久久久久亚洲调教| 国产在线精品播放| 成人av在线天堂| 91av免费观看91av精品在线| 日韩在线观看免费高清完整版| 亚洲精品永久免费精品| 亚洲美女av在线播放| 国产免费一区二区三区在线观看| 国产精品视频不卡| 中文字幕日韩欧美在线| 亚洲欧美综合精品久久成人| 亚洲老司机av| 欧美专区在线观看| 欧美精品久久久久久久免费观看| 亚洲精品456在线播放狼人| 欧美另类极品videosbestfree| 午夜精品一区二区三区视频免费看| 色综合久久久久久中文网| 亚洲在线免费看| 久久久中精品2020中文| 97色在线观看| 午夜美女久久久久爽久久| 欧美最猛性xxxxx(亚洲精品)| 亚洲国产成人91精品| 国产精品丝袜一区二区三区| 中文字幕亚洲综合久久| 国产精品免费网站| 97人人做人人爱| 久久五月天色综合| 美女av一区二区三区| 亚洲欧美一区二区三区在线| 4388成人网| 亚洲视频视频在线| 久99九色视频在线观看| 亚洲欧美在线一区二区| 欧美激情精品久久久久| 亚洲毛茸茸少妇高潮呻吟| 欧美情侣性视频| 欧美成人性色生活仑片| 亚洲美女在线视频| 97色在线播放视频| 亚洲人成电影在线播放| 97免费视频在线播放| 亚洲欧洲第一视频| 成人久久久久爱| 久久久久日韩精品久久久男男| 亚洲韩国欧洲国产日产av| 亚洲第一网站免费视频| 日韩在线观看免费全| 国产一区玩具在线观看| 国内免费久久久久久久久久久| 国产精品视频xxx| 精品久久久久久久久久| 日韩影视在线观看| 久久久91精品| 国产精品三级在线| 亚洲精品第一国产综合精品| 97精品国产aⅴ7777| 一区二区欧美日韩视频| 欧美精品videos性欧美| 国产精品对白刺激| 隔壁老王国产在线精品| 国产美女高潮久久白浆| 亚洲精品一区中文字幕乱码| 亚洲欧美综合区自拍另类| 精品欧美国产一区二区三区| 91日本在线观看| 日韩中文字幕国产精品| 久久在精品线影院精品国产| 555www成人网| 欧美wwwxxxx| 国产日韩精品一区二区| 亚洲男人天堂2019| 亚洲午夜久久久影院| 不卡av电影在线观看| 黑人狂躁日本妞一区二区三区| 日韩一区二区在线视频| 国产精品一区二区三区久久久| 国产精品电影在线观看| 亚洲欧美激情四射在线日| 久久久久久高潮国产精品视| 久久影院资源站| 久久久免费观看| xxxxxxxxx欧美| 日韩av免费在线播放| 欧美最猛性xxxxx亚洲精品| 亚洲欧美资源在线| 日韩有码在线电影| 精品久久久国产| 欧美日韩国产综合视频在线观看中文| 另类天堂视频在线观看| 久久久女人电视剧免费播放下载| 欧美另类暴力丝袜| 国产精品第一视频| 国产精品日韩在线| 欧美精品做受xxx性少妇| 蜜臀久久99精品久久久久久宅男| 97人洗澡人人免费公开视频碰碰碰| 一个人看的www久久| 亚洲欧美制服综合另类| 在线电影欧美日韩一区二区私密| 91欧美激情另类亚洲| 国产在线精品一区免费香蕉| zzjj国产精品一区二区| 91网站免费观看| 国产亚洲视频中文字幕视频|