軟體設計模式學習(十五)享元模式
阿新 • • 發佈:2020-05-09
> 當系統中存在大量相同或相似的物件時,享元模式是一種較好的解決方案,它通過共享技術實現相同或相似的細粒度物件的複用,從而節約記憶體空間。享元模式提供了一個享元池用於儲存已經建立好的享元物件,並通過享元工廠類將享元物件提供給客戶端使用。
## 模式動機 使用面向物件技術開發時,很多情況下需要在系統中增加類和物件的個數,並且這些物件有些是相同或相似的。當物件太多時,將導致執行代價過高,效能下降等問題。為了避免系統中出現大量相同或相似的物件,享元模式通過共享技術實現相同或相似物件的重用,相同的物件都指向一個例項,儲存這個例項的物件稱為享元池。
## 模式設計 系統中有些物件並不完全相同,而只是相似,因此需要先找出這些物件的共同點,在享元類中封裝這些共同的內容。不同的內容可以通過外部應用程式來設定,而不進行共享,在享元模式中可以共享的相同內容稱為內部狀態,而那些需要外部環境設定的不能共享的內容稱為外部狀態。 在享元模式中通常會出現工廠模式,需要建立一個享元工廠來維護一個享元池,用於儲存具有相同內部狀態的享元物件。實際使用中,能夠共享的內部狀態是有限的,因此享元物件一般都設計為較小的物件,它所包含的內部狀態較少,這種狀態一般稱為細粒度物件。享元模式的目的就是使用共享技術來實現大量細粒度物件的複用。
## 模式結構 ![在這裡插入圖片描述](https://img-blog.csdnimg.cn/20200509175332969.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L0NTRE5faGFuZHNvbWU=,size_16,color_FFFFFF,t_70) 1. Flyweight(抽象享元類) 抽象享元類宣告是一個介面,通過它可以接受並作用於外部狀態。在抽象享元類定義了具體享元類公共的方法,這些方法可以向外界提供享元物件的內部資料(內部狀態),同時也可以通過這些方法來設定外部資料(外部狀態) 2. ConcreteFlyweight(具體享元類) 具體享元類實現了抽象享元介面,儲存了內部狀態,具體享元物件是可以共享的。可以結合單例模式來設計享元具體類,為每一個具體享元類提供唯一的享元物件。 3. UnsharedConcreteFlyweight(非共享具體享元類) 不能被共享的抽象享元類的子類被設計為非共享具體享元類。當需要一個非共享具體享元類的物件時可以直接通過例項化建立。在某些享元模式的層次結構中,非共享具體享元物件還可以將具體享元物件作為子節點。 4. FlyweightFactory(享元工廠類) 享元工廠類用於建立並管理享元物件,它針對抽象享元類程式設計,將各種型別的具體享元物件儲存在一個享元池中。當用戶請求一個具體享元物件時,享元工廠提供一個儲存在享元池已建立的例項或者建立一個例項,返回該新建立的例項並將其儲存在享元池中。
## 模式優缺點 優點如下: 1. 極大減少記憶體中物件的數量 2. 享元模式的外部狀態相對獨立,不會影響其內部狀態,因此享元物件可以在不同的環境中被共享。 缺點如下: 1. 享元模式是系統更加複雜,需要分離出內部狀態和外部狀態。 2. 讀取外部狀態會使執行時間變長。
## 模式適用環境 以下情況可以使用享元模式: 1. 一個系統有大量相同或相似物件,這類物件的大量使用造成記憶體的大量耗費 2. 物件的大部分狀態都可外部化,可以將這些外部狀態傳入物件中。 3. 維護享元池需要耗費資源,因此應當在多次重複使用享元物件時才值得使用享元模式
## 單純享元模式 即所有抽象享元類的子類都可以共享,不存在非共享具體享元類 ![在這裡插入圖片描述](https://img-blog.csdnimg.cn/20200509175525322.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L0NTRE5faGFuZHNvbWU=,size_16,color_FFFFFF,t_70)
## 複合享元模式 將單純享元模式與組合模式加以組合,可以形成複合享元物件。這樣的複合享元物件本身不能共享,但它們可以分解成單純享元物件,而後者可以共享。通過複合享元模式,可以確保複合享元類 CompositeConcreteFlyweight 中所包含的每個單純享元類 ConcreteFlyweight 都具有相同的外部狀態,而這些單純享元的內部狀態往往不同。 ![在這裡插入圖片描述](https://img-blog.csdnimg.cn/20200509175546320.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L0NTRE5faGFuZHNvbWU=,size_16,color_FFFFFF,t_70)