橋接模式 VS 裝飾器模式碌补、狀態(tài)模式 VS 策略模式的微妙之處
Foundations of Software Engineering 的 Design Pattern 學(xué)習(xí)筆記整理
關(guān)鍵詞:設(shè)計模式壁公、橋接模式洪添、裝飾器模式、狀態(tài)模式掌唾、策略模式( Design Pattern, Bridge Pattern, Decorator Pattern, State Pattern, Strategy Pattern)
本文簡述了橋接模式(Bridge Pattern)和裝飾器模式(Decorator Pattern)放前,以及狀態(tài)模式(State Pattern)和策略模式(Strategy Pattern)的微妙之處,以及它們之間是如何“看起來就像另一者的”糯彬。
橋接是指能夠沿著不同的維度(along distinct dimensions)以多種方式(in more than one way)來指定一個對象凭语,通常使用子類型(sub-typing)和注入(injection)
new BoldText("blah blah", new UTF16Encoding))
new ItalicsText("blah blah", new ASCIIEncoing))
// where BoldText and ItalicsText are subclasses of Text and UTF16Encoding and ASCIIEncoding are subclasses of TextEncoding
裝飾器是指能夠以一種任意的方式向一個對象添加特征(responsibilities, embellishments, or features),通常使用包裝(wrapping)
裝飾器所描述的是一種 specialization情连,但是有著截然不同的機制叽粹,并且可以創(chuàng)建的變化不一定是不同維度的(variations you can create are not points on distinct dimensions):在同一維度上,多個特征可以同時添加到一個對象上却舀。使用裝飾器模式需要依賴?yán)^承虫几,但是子類型卻并不特殊化目標(biāo)對象(subtyping does not specialize the target object),我們可以使用包裝來完成這個事情(injection it into a higher level object (wrapper) does the trick)挽拔。
new Underlined(new Italics(new Bold(new Text("blah blah")))).
在上面這個例子中辆脸,我們可以使用裝飾器模式將 blah blah
變成斜體的、加粗的螃诅、帶下劃線的表示啡氢,但是不能使用橋接模式將這段文本變成既是加粗又是斜體的,因為 Bold 和 Italics 都是同一個維度下的分化(specializations along the style dimension)术裸,你必須選擇其中之一倘是。同樣,UTF16 和 ASCII 是編碼維度下的不同分化袭艺,你也必須選擇其中之一搀崭。
如果你想用橋接模式創(chuàng)建一個既是粗體、又是斜體的風(fēng)格猾编,那么你就必須定義一個名為 BoldAndItalicsText
的 Text 的子類瘤睹,這顯得不合理是嗎?是的答倡,所以轰传,你就會意識到,粗體瘪撇、斜體當(dāng)然是可以被一起使用的获茬,因為它們是特征港庄,而不是單一維度下面的不同分化,所以你需要的其實是裝飾器模式锦茁,而不是橋接模式攘轩。
再來回顧一下這個使用橋接模式的經(jīng)典例子:在一個維度上,我們有 NoSQL DB码俩、SQL DB 和 Mock DB度帮,它們是單一維度下的不同分化,而在另一個維度上稿存,我們有 Backlog DB 和 Product DB 等笨篷。
狀態(tài)模式和策略模式意外地有著類似的類圖,但是它們的意圖是不同的瓣履。
狀態(tài)不僅僅表達了在運行時去改變一個對象的行為率翅,它更加強調(diào)的是一個對象能夠識別它自己的內(nèi)部狀態(tài)并相應(yīng)地改變它的行為(也可能是改變它的狀態(tài),因此需要在對象內(nèi)部實現(xiàn)一個狀態(tài)機)袖迎。
作為策略模式的典型例子冕臭,我們會在一個機器人對象中注入一個不同的防撞策略,這時我們并沒有更改機器人的內(nèi)部狀態(tài)燕锥,所以這僅僅是選了不同的策略辜贵。當(dāng)我們在一個項目經(jīng)理對象中注入一個不同的報告生成過濾器時,我們沒有改變內(nèi)部狀態(tài)归形,只是改變了在任何狀態(tài)下的報告的打印策略托慨。
但是,如果需要一臺自動售貨機在有足夠的錢存入時與錢不夠時表現(xiàn)不同暇榴,這就是關(guān)于狀態(tài)的了厚棵。自動售貨機在有足夠資金的時候會切換狀態(tài),并且在每個狀態(tài)下只能執(zhí)行某些行為蔼紧。
因此婆硬,意圖在設(shè)計模式中是很重要的。不同的意圖意味著奸例,不同的選擇柿祈。