1.定義
策略模式:定義了算法簇[cù]擂啥,分別封裝起來(lái),讓它們之間可以相互替換,此模式讓算法的變化獨(dú)立于使用算法的客戶背零。
2.背景
(本來(lái)是打算用自己業(yè)務(wù)中的某個(gè)場(chǎng)景來(lái)進(jìn)行講解的,很遺憾之前并沒(méi)有使用過(guò)該模式无埃,之后會(huì)嘗試使用)
Joe上班的公司做了一套相當(dāng)成功的模擬鴨子游戲徙瓶;SimUDuck。游戲中會(huì)出現(xiàn)各種鴨子嫉称,一邊游泳戲水侦镇,一邊呱呱叫。此系統(tǒng)的內(nèi)部設(shè)計(jì)使用了標(biāo)準(zhǔn)的OO技術(shù)织阅,設(shè)計(jì)了一個(gè)鴨子的超類(lèi)(Superclass)壳繁,并讓各種鴨子繼承此超類(lèi)。但是為了提高競(jìng)爭(zhēng)力荔棉,主管們確定闹炉,此模擬程序需要會(huì)飛的鴨子來(lái)將競(jìng)爭(zhēng)者拋在后面。
3.方案
1.Joe首先想到的方法是在Duck類(lèi)中加上fly()方法润樱,然后所有鴨子都會(huì)繼承fly()渣触。但是這個(gè)會(huì)出現(xiàn)所有的鴨子都有了飛的功能(包括“橡皮鴨子”),他體會(huì)到了一件事:當(dāng)涉及“維護(hù)”時(shí)壹若,為了“復(fù)用”目的而使用繼承嗅钻,結(jié)局并不完美。
2.Joe之后想到的是舌稀,繼續(xù)采用在Duck中添加fly()方法啊犬,可以把橡皮鴨中的fly()方法覆蓋掉,可是以后我們加入誘餌鴨壁查,又會(huì)如何觉至?誘餌鴨是木頭假鴨,不會(huì)飛也不會(huì)叫睡腿。顯然這并不是一個(gè)好的解決方案语御,每當(dāng)有新的鴨子子類(lèi)出現(xiàn)峻贮,他就要被迫檢查并可能需要覆蓋fly()。
現(xiàn)在我們知道使用繼承并不能很好地解決問(wèn)題应闯,因?yàn)轼喿拥男袨樵谧宇?lèi)里不斷地改變纤控,并且讓所有的子類(lèi)都有這些行為是不恰當(dāng)?shù)摹lyable接口似乎是個(gè)不錯(cuò)的方法碉纺,但是Java接口不具有實(shí)現(xiàn)代碼船万,所以繼承接口無(wú)法達(dá)到代碼的復(fù)用。幸運(yùn)的是骨田,有一個(gè)原則耿导,恰好適用此狀況。
設(shè)計(jì)原則:找出應(yīng)用中可能需要變化之處态贤,把它們獨(dú)立出來(lái)舱呻,不要和那些不需要變化的代碼混在一起。
我們知道Duck類(lèi)內(nèi)的fly() 飛 和quack() 呱呱叫 會(huì)隨著鴨子的不同而改變悠汽,為了要把這個(gè)兩個(gè)行為從Duck類(lèi)中分開(kāi)箱吕,我們將它們從Duck類(lèi)中取出來(lái),建立一組新類(lèi)型來(lái)代表每個(gè)行為柿冲。我們希望一切能有彈性茬高。畢竟,正是因?yàn)橐婚_(kāi)始鴨子行為沒(méi)有彈性姻采,才讓我們走上現(xiàn)在這條路雅采。我們還想能夠“指定”行為到鴨子的實(shí)例。比方說(shuō)慨亲,我們想要產(chǎn)生一個(gè)新的綠頭鴨實(shí)例,并指定特定“類(lèi)型”的飛行行為給它宝鼓。干脆順便讓鴨子的行為可以動(dòng)態(tài)地改變好了刑棵。換句話說(shuō),我們應(yīng)該在鴨子類(lèi)中包含設(shè)定行為的方法愚铡,這樣就可以在“運(yùn)行時(shí)動(dòng)態(tài)地改變綠頭鴨的飛行行為蛉签。
設(shè)計(jì)原則:針對(duì)接口編程(“針對(duì)接口編程”真正的意思是“針對(duì)超類(lèi)型supertype編程”),而不是針對(duì)實(shí)現(xiàn)編程沥寥。
這樣鴨子就有一個(gè)飛行為碍舍,有一個(gè)呱呱叫行為。有一個(gè)關(guān)系相當(dāng)有趣:每一鴨子都有一個(gè)FlyBehavior和一個(gè)QuackBehavior邑雅,好將飛行和呱呱叫委托給它們代為處理片橡。
3.最后的設(shè)計(jì)方案為,我們?cè)贒uck類(lèi)中淮野,創(chuàng)建接口FlyBehavior和QuackBehavior的實(shí)例捧书,進(jìn)行操作吹泡,將飛行和呱呱叫委托給上面的兩個(gè)接口處理。
當(dāng)你將兩個(gè)或多個(gè)類(lèi)結(jié)合起來(lái)使用经瓷,如同比例一般爆哑,這就是組合。這種做法和“繼承”不同的地方在于舆吮,鴨子的行為不是繼承來(lái)的揭朝,而是和適當(dāng)?shù)男袨閷?duì)象組合來(lái)的。
設(shè)計(jì)原則:多用組合色冀,少用繼承潭袱。
這第三種方案就是應(yīng)用策略模式,定義了算法簇呐伞,分別封裝起來(lái)敌卓,讓它們之間可以互相替換,此模式讓算法的變化獨(dú)立于使用算法的客戶伶氢。