設計模式之 十一 享元模式Flyweight

2021-05-27 07:30:20 字數 2787 閱讀 5253

flyweight模式定義:

避免大量擁有相同內容的小類的開銷(如耗費記憶體),使大家共享乙個類(元類).

為什麼使用?

物件導向語言的原則就是一切都是物件,但是如果真正使用起來,有時物件數可能顯得很龐大,比如,字處理軟體,如果以每個文字都作為乙個物件,幾千個字,物件數就是幾千,無疑耗費記憶體,那麼我們還是要"求同存異",找出這些物件群的共同點,設計乙個元類,封裝可以被共享的類,另外,還有一些特性是取決於應用(context),是不可共享的,這也flyweight中兩個重要概念內部狀態intrinsic和外部狀態extrinsic之分.

說白點,就是先捏乙個的原始模型,然後隨著不同場合和環境,再產生各具特徵的具體模型,很顯然,在這裡需要產生不同的新物件,所以flyweight模式中常出現factory模式.flyweight的內部狀態是用來共享的,flyweight factory負責維護乙個flyweight pool(模式池)來存放內部狀態的物件.

flyweight模式是乙個提高程式效率和效能的模式,會大大加快程式的執行速度.應用場合很多:比如你要從乙個資料庫中讀取一系列字串,這些字串中有許多是重複的,那麼我們可以將這些字串儲存在flyweight池(pool)中.

如何使用?

我們先從flyweight抽象介面開始:

public inte***ce flyweight 

//用於本模式的抽象資料型別(自行設計)

public inte***ce extrinsicstate

下面是介面的具體實現(concreteflyweight) ,並為內部狀態增加記憶體空間, concreteflyweight必須是可共享的,它儲存的任何狀態都必須是內部(intrinsic),也就是說,concreteflyweight必須和它的應用環境場合無關.

public class concreteflyweight implements flyweight

當然,並不是所有的flyweight具體實現子類都需要被共享的,所以還有另外一種不共享的concreteflyweight:

public class unsharedconcreteflyweight implements flyweight

flyweight factory負責維護乙個flyweight池(存放內部狀態),當客戶端請求乙個共享flyweight時,這個factory首先搜尋池中是否已經有可適用的,如果有,factory只是簡單返回送出這個物件,否則,建立乙個新的物件,加入到池中,再返回送出這個物件.池

public class flyweightfactory  

return flyweight; } }

至此,flyweight模式的基本框架已經就緒,我們看看如何呼叫:

flyweightfactory factory = new flyweightfactory(); 

flyweight fly1 = factory.getflyweight( "fred" ); 

flyweight fly2 = factory.getflyweight( "wilma" );

......

從呼叫上看,好象是個純粹的factory使用,但奧妙就在於factory的內部設計上.

flyweight模式在xml等資料來源中應用

我們上面已經提到,當大量從資料來源中讀取字串,其中肯定有重複的,那麼我們使用flyweight模式可以提高效率,以唱片cd為例,在乙個xml檔案中,存放了多個cd的資料.

每個cd有三個字段:

1.出片日期(year)

2.歌唱者姓名等資訊(artist)

3.唱片曲目 (title)

其中,歌唱者姓名有可能重複,也就是說,可能有同乙個演唱者的多個不同時期 不同曲目的cd.我們將"歌唱者姓名"作為可共享的concreteflyweight.其他兩個字段作為unsharedconcreteflyweight.

首先看看資料來源xml檔案的內容:

<?xml version="1.0"?>

another green world

1978

eno, brian

greatest hits

1950

holiday, billie

taking tiger mountain (by strategy)

1977

eno, brian

雖然上面舉例cd只有3張,cd可看成是大量重複的小類,因為其中成分只有三個字段,而且有重複的(歌唱者姓名).

cd就是類似上面介面 flyweight:

public class cd

public int getyear()

public artist getartist()

public void settitle(string t)

public void setyear(int y)

public void setartist(artist a)

}將"歌唱者姓名"作為可共享的concreteflyweight:

public class artist

artist(string n) 

}再看看flyweight factory,專門用來製造上面的可共享的concreteflyweight:artist

public class artistfactory

return result;}}

當你有幾千張甚至更多cd時,flyweight模式將節省更多空間,共享的flyweight越多,空間節省也就越大.

**

設計模式十一之享元模式

2.模式的結構與實現 3.模式在開源軟體中的應用 在物件導向程式設計過程中,有時會面臨要建立大量相同或相似物件例項的問題。建立那麼多的物件將會耗費很多的系統資源,它是系統效能提高的乙個瓶頸。這些物件有很多相似的地方,如果能把它們相同的部分提取出來共享,則能節省大量的系統資源,這就是享元模式的產生背景...

設計模式(十一)之 Flyweight 享元

避免大量擁有相同內容的小類的開銷 如耗費記憶體 使大家共享乙個類 元類 物件導向語言的原則就是一切都是物件,但是如果真正使用起來,有時物件數可能顯得很龐 大,比如,字處理軟體,如果以每個文字都作為乙個物件,幾千個字,物件數就是幾千,無疑耗 費記憶體,那麼我們還是要 求同存異 找出這些物件群的共同點,...

C C 設計模式 十一 享元模式

flyweight模式也叫享元模式,是構造型模式之一,它通過與其他類似物件共享資料來減小記憶體占用。在物件導向系統的設計何實現中,建立物件是最為常見的操作。這裡面就有乙個問題 如果乙個應用程式使用了太多的物件,就會造成很大的儲存開銷。特別是對於大量輕量級 細粒度 的物件,比如在文件編輯器的設計過程中...