4、HeadFirst--策略模式(Swift)

1.概述

在軟件開(kāi)發(fā)中也常常遇到類(lèi)似的情況额获,實(shí)現(xiàn)某一個(gè)功能有多種算法或者策略够庙,我們可以根據(jù)環(huán)境或者條件的不同
選擇不同的算法或者策略來(lái)完成該功能。如查找抄邀、排序等耘眨,一種常用的方法是硬編碼(Hard Coding)在一個(gè)類(lèi)中,
如需要提供多種查找算法境肾,可以將這些算法寫(xiě)到一個(gè)類(lèi)中剔难,在該類(lèi)中提供多個(gè)方法,
每一個(gè)方法對(duì)應(yīng)一個(gè)具體的查找算法奥喻;當(dāng)然也可以將這些查找算法封裝在一個(gè)統(tǒng)一的
方法中偶宫,通過(guò)if…else…或者case等條件判斷語(yǔ)句來(lái)進(jìn)行選擇。這兩種實(shí)現(xiàn)方法我
們都可以稱(chēng)之為硬編碼环鲤,如果需要增加一種新的查找算法纯趋,需要修改封裝算法類(lèi)
的源代碼;更換查找算法冷离,也需要修改客戶(hù)端調(diào)用代碼吵冒。在這個(gè)算法類(lèi)中封裝了
大量查找算法,該類(lèi)代碼將較復(fù)雜西剥,維護(hù)較為困難痹栖。如果我們將這些策略包含在
客戶(hù)端,這種做法更不可取瞭空,將導(dǎo)致客戶(hù)端程序龐大而且難以維護(hù)揪阿,如果存在大
量可供選擇的算法時(shí)問(wèn)題將變得更加嚴(yán)重疗我。

demo地址,歡迎star

例子1:

商場(chǎng)產(chǎn)品銷(xiāo)售:可以正常價(jià)格售出南捂,可以打折形式售出碍粥,也可以通過(guò)積分的形式,不管哪種形式黑毅,最后要的就是一個(gè)最后的總價(jià)

例子2:

出行旅游:我們可以有幾個(gè)策略可以考慮:可以騎自行車(chē),汽車(chē)钦讳,做火車(chē)矿瘦,飛 機(jī)。每個(gè)策略都可以得到相同的結(jié)果愿卒,但是它們使用了不同的資源缚去。選擇策略的依據(jù)是費(fèi)用,時(shí)間琼开,使用工具還有每種方式的方便程度 易结。

例子3:

HeadFirst案例:不同的鴨子具有不同的飛行行為,鳴叫行為柜候,不同的飛行行為或鳴叫行為就是不同的算法策略

2.問(wèn)題

如何讓算法和對(duì)象(客戶(hù)端)分開(kāi)來(lái)搞动,使得算法可以獨(dú)立于使用它的客戶(hù)而變化?

3.解決方案

策略模式:
定義一系列的算法,把每一個(gè)算法封裝起來(lái), 并且使它們可相互替換渣刷。本模式使得算法可獨(dú)立于使用它的客戶(hù)而變化鹦肿。也稱(chēng)為政策模式(Policy)。 策略模式把對(duì)象本身和運(yùn)算規(guī)則區(qū)分開(kāi)來(lái)辅柴,其功能非常強(qiáng)大箩溃,因?yàn)檫@個(gè)設(shè)計(jì)模式本身的核心思想就是面向?qū)ο缶幊痰亩嘈涡缘乃枷搿?/code>

4.適用性

當(dāng)存在以下情況時(shí)使用Strategy模式

  • 許多相關(guān)的類(lèi)僅僅是行為有異。 “策略”提供了一種用多個(gè)行為中的一個(gè)行為來(lái)配置一個(gè)類(lèi)的方法碌嘀。即一個(gè)系統(tǒng)需要?jiǎng)討B(tài)地在幾種算法中選擇一種涣旨。
  • 需要使用一個(gè)算法的不同變體。例如股冗,你可能會(huì)定義一些反映不同的空間 /時(shí)間權(quán)衡的算法霹陡。當(dāng)這些變體實(shí)現(xiàn)為一個(gè)算法的類(lèi)層次時(shí) ,可以使用策略模式。
  • 算法使用客戶(hù)不應(yīng)該知道的數(shù)據(jù)魁瞪∧侣桑可使用策略模式以避免暴露復(fù)雜的、與算法相關(guān)的數(shù)據(jù)結(jié)構(gòu)导俘。
  • 一個(gè)類(lèi)定義了多種行為 , 并且這些行為在這個(gè)類(lèi)的操作中以多個(gè)條件語(yǔ)句的形式出現(xiàn)峦耘。將相關(guān)的條件分支移入它們各自的Strategy類(lèi)中以代替這些條件語(yǔ)句。

5.swift實(shí)現(xiàn)結(jié)構(gòu)圖

swift策略模式.png

6.模式的組成

  • 環(huán)境類(lèi)(Context):用一個(gè)ConcreteStrategy對(duì)象來(lái)配置旅薄。維護(hù)一個(gè)對(duì)Strategy對(duì)象的引用辅髓∑溃可定義一個(gè)接口來(lái)讓Strategy訪(fǎng)問(wèn)它的數(shù)據(jù)。
  • 抽象策略類(lèi)(Strategy):定義所有支持的算法的公共接口洛口。 Context使用這個(gè)接口來(lái)調(diào)用某ConcreteStrategy定義的算法矫付。
  • 具體策略類(lèi)(ConcreteStrategy):以Strategy接口實(shí)現(xiàn)某具體算法。

7.效果

Strategy模式有下面的一些優(yōu)點(diǎn):

  • 相關(guān)算法系列 Strategy類(lèi)層次為Context定義了一系列的可供重用的算法或行為第焰。 繼承有助于析取出這些算法中的公共功能买优。
  • 擴(kuò)展性良好: 繼承提供了另一種支持多種算法或行為的方法。你可以直接生成一個(gè)Context類(lèi)的子類(lèi)挺举,從而給它以不同的行為杀赢。但這會(huì)將行為硬行編制到 Context中,而將算法的實(shí)現(xiàn)與Context的實(shí)現(xiàn)混合起來(lái),從而使Context難以理解湘纵、難以維護(hù)和難以擴(kuò)展脂崔,而且還不能動(dòng)態(tài)地改變算法。最后你得到一堆相關(guān)的類(lèi) , 它們之間的唯一差別是它們所使用的算法或行為梧喷。 將算法封裝在獨(dú)立的Strategy類(lèi)中使得你可以獨(dú)立于其Context改變它砌左,使它易于切換、易于理解铺敌、易于擴(kuò)展汇歹。
  • 避免使用多重條件判斷 :Strategy模式提供了用條件語(yǔ)句選擇所需的行為以外的另一種選擇。當(dāng)不同的行為堆砌在一個(gè)類(lèi)中時(shí) ,很難避免使用條件語(yǔ)句來(lái)選擇合適的行為偿凭。將行為封裝在一個(gè)個(gè)獨(dú)立的Strategy類(lèi)中消除了這些條件語(yǔ)句秤朗。含有許多條件語(yǔ)句的代碼通常意味著需要使用Strategy模式。
  • 算法可以自由切換:實(shí)現(xiàn)的選擇 Strategy模式可以提供相同行為的不同實(shí)現(xiàn)笔喉。客戶(hù)可以根據(jù)不同時(shí)間 /空間權(quán)衡取舍要求從不同策略中進(jìn)行選擇取视。
  • 降低耦合:策略以相同的方式調(diào)用所有的算法,減少客戶(hù)端與算法類(lèi)之間的耦合

Strategy模式缺點(diǎn):

  • 所有策略類(lèi)都需要對(duì)外暴露: 本模式有一個(gè)潛在的缺點(diǎn)常挚,就是一個(gè)客戶(hù)要選擇一個(gè)合適的Strategy就必須知道這些Strategy到底有何不同作谭。此時(shí)可能不得不向客戶(hù)暴露具體的實(shí)現(xiàn)問(wèn)題。因此僅當(dāng)這些不同行為變體與客戶(hù)相關(guān)的行為時(shí) , 才需要使用Strategy模式奄毡。
  • Strategy和Context之間的通信開(kāi)銷(xiāo) :無(wú)論各個(gè)ConcreteStrategy實(shí)現(xiàn)的算法是簡(jiǎn)單還是復(fù)雜, 它們都共享Strategy定義的接口折欠。因此很可能某些 ConcreteStrategy不會(huì)都用到所有通過(guò)這個(gè)接口傳遞給它們的信息;簡(jiǎn)單的 ConcreteStrategy可能不使用其中的任何信息吼过!這就意味著有時(shí)Context會(huì)創(chuàng)建和初始化一些永遠(yuǎn)不會(huì)用到的參數(shù)锐秦。如果存在這樣問(wèn)題 , 那么將需要在Strategy和Context之間更進(jìn)行緊密的耦合。
  • 策略類(lèi)會(huì)增多:可以通過(guò)使用享元模式在一定程度上減少對(duì)象的數(shù)量盗忱。 增加了對(duì)象的數(shù)目 Strategy增加了一個(gè)應(yīng)用中的對(duì)象的數(shù)目酱床。有時(shí)你可以將 Strategy實(shí)現(xiàn)為可供各Context共享的無(wú)狀態(tài)的對(duì)象來(lái)減少這一開(kāi)銷(xiāo)。任何其余的狀態(tài)都由 Context維護(hù)趟佃。Context在每一次對(duì)Strategy對(duì)象的請(qǐng)求中都將這個(gè)狀態(tài)傳遞過(guò)去扇谣。共享的 Strategy不應(yīng)在各次調(diào)用之間維護(hù)狀態(tài)昧捷。

8.實(shí)現(xiàn)

抽象類(lèi)

//這里其實(shí)可以用協(xié)議替代
class CashSuper: NSObject {
    //優(yōu)點(diǎn)1: 繼承有助于析取出這些算法中的公共功能。
    //缺點(diǎn)2: Strategy和Context之間的通信開(kāi)銷(xiāo)罐寨,對(duì)于子類(lèi)實(shí)現(xiàn)的通信接口(acceptCash)靡挥,參數(shù)(money)可以會(huì)永遠(yuǎn)不會(huì)使用
    func acceptCash(money: Double) -> Double {
        return 0.0
    }
}

具體實(shí)現(xiàn)對(duì)象--打折類(lèi)(其它策略類(lèi) 類(lèi)似)

//打折類(lèi)
class CashRebate: CashSuper {
    
    private var moneyRebate: Double = 1.0
    
    init(moneyRebate: Double) {
        self.moneyRebate = moneyRebate
        super.init()
    }

    //打折返回
    override func acceptCash(money: Double) -> Double {
        return money * moneyRebate
    }

}

客戶(hù)端對(duì)象

//客戶(hù)端對(duì)象(具體使用的對(duì)象)
class CashContext: NSObject {

    private var cs: CashSuper!
    
    //優(yōu)點(diǎn)4: 客戶(hù)可以根據(jù)不同時(shí)間 /空間權(quán)衡取舍要求從不同策略中進(jìn)行選擇。
    //優(yōu)點(diǎn)3: 將算法封裝在獨(dú)立的Strategy類(lèi)中使得你可以獨(dú)立于其Context改變它鸯绿,使它易于切換跋破、易于理解、易于擴(kuò)展瓶蝴。
    //缺點(diǎn)3: 策略模式將造成產(chǎn)生很多策略類(lèi)
    func cashContext(type: CacultorType) {
        switch type {
        case .Normal:
            cs = CashNormal()
        case .Rebate:
            cs = CashRebate(moneyRebate: 0.8)
        case .Return:
            cs = CashReturn(moneyCondition: 300, moneyReturn: 100)
        }
    }
    
    //優(yōu)點(diǎn)2: 策略以相同的方式調(diào)用所有的算法幔烛,減少客戶(hù)端與算法類(lèi)之間的耦合
    //傳入初始金額,利用私有cs囊蓝,計(jì)算最終金額
    func getResult(money: Double) -> Double {
        return cs.acceptCash(money: money)
    }
}

demo地址,歡迎star

9.HeadFirst -- 策略模式

class Duck {

    //fly 和 quack 不是每個(gè)子類(lèi)都可能具備的所以它們是變化的部分令蛉,應(yīng)該抽取出來(lái)
    //設(shè)計(jì)原則:使用組合(FlyBehavior, QuackBehivor)而不是使用繼承
    var flyBehavior: FlyBehavior?        //針對(duì)接口(FlyBehavior)編程而不是針對(duì)實(shí)現(xiàn)編程
    var quackBehavior: QuackBehivor?     //針對(duì)接口(QuackBehivor)編程而不是針對(duì)實(shí)現(xiàn)編程

    //不需要變化的行為
    func swim() {
        print("I can swim")
    }

    //根據(jù)子類(lèi)的而變化的行為
    func display() {

    }

    func performQuack() {
        flyBehavior?.fly()
    }

    func performFly() {
        quackBehavior?.quack()
    }
}

//具體子類(lèi)聚霜,綠頭鴨
class Mallard: Duck {
    override func display() {
        print("一只綠頭鴨")
    }
}

變化的行為Fly

//設(shè)計(jì)原則:封裝變化-找出應(yīng)用中需要變化的地方,將他們獨(dú)立出來(lái)珠叔,不要和不需要變化的代碼混合到一塊
//設(shè)計(jì)原則:針對(duì)接口編程而不是針對(duì)實(shí)現(xiàn)編程

protocol FlyBehavior {
    func fly()
}

extension FlyBehavior {
    func fly() {
        print("默認(rèn)實(shí)現(xiàn)")
    }
}

//下面是針對(duì)FlyBehavior協(xié)議定義的算法簇
class FlyWithWings: FlyBehavior {
    func fly() {
        print("I can fly!")
    }
}

class FlyNoWay: FlyBehavior {
//    func fly() {
//        print("I can't fly!")
//    }
}

變化的行為Quack

//設(shè)計(jì)原則:封裝變化-找出應(yīng)用中需要變化的地方蝎宇,將他們獨(dú)立出來(lái),不要和不需要變化的代碼混合到一塊
//設(shè)計(jì)原則:針對(duì)接口編程而不是針對(duì)實(shí)現(xiàn)編程
protocol QuackBehivor {
    func quack()
}

//下面是針對(duì)QuackBehivor協(xié)議定義的算法簇
class Quack: QuackBehivor {
    func quack() {
        print("Quack")
    }
}

class Slient: QuackBehivor {
    func quack() {
        print("slient")
    }
}

class Squeak: QuackBehivor {
    func quack() {
        print("Squeak")
    }
}

總結(jié)

HeadFirst策略模式使用的設(shè)計(jì)原則

  • 封裝變化-找出應(yīng)用中需要變化的地方祷安,將他們獨(dú)立出來(lái)姥芥,不要和不需要變化的代碼混合到一塊
  • 針對(duì)接口編程而不是針對(duì)實(shí)現(xiàn)編程-針對(duì)接口編程實(shí)際上就是針對(duì)超類(lèi)型編程,其關(guān)鍵在于多態(tài)汇鞭,利用多態(tài)就可以實(shí)現(xiàn)超類(lèi)型編程凉唐,也就是根據(jù)實(shí)現(xiàn)情況來(lái)執(zhí)行具體的行為,而不是綁定在超類(lèi)型的行為上霍骄。通俗一點(diǎn)來(lái)說(shuō)台囱,就是聲明一個(gè)變量,這個(gè)變量的類(lèi)型就是超類(lèi)型(超類(lèi)型通過(guò)是一個(gè)抽象類(lèi)或一個(gè)接口读整,在swift中接口就是協(xié)議,不過(guò)和java的接口相比簿训,swift協(xié)議可以有默認(rèn)實(shí)現(xiàn)),只要具體實(shí)現(xiàn)此超類(lèi)型的對(duì)象都可以直接賦值給上面的變量。也就是說(shuō)不用關(guān)心執(zhí)行具體的對(duì)象類(lèi)型米间。
  • 多使用組合而不是繼承

HeadFirst策略模式

  • 策略模式定義了算法族强品,分別封裝起來(lái),讓他們之前可以相互替換屈糊。此模式讓算法的變化獨(dú)立于使用算法的客戶(hù)的榛。

demo地址,歡迎star

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末逻锐,一起剝皮案震驚了整個(gè)濱河市困曙,隨后出現(xiàn)的幾起案子表伦,更是在濱河造成了極大的恐慌,老刑警劉巖慷丽,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蹦哼,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡要糊,警方通過(guò)查閱死者的電腦和手機(jī)纲熏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)锄俄,“玉大人局劲,你說(shuō)我怎么就攤上這事∧淘” “怎么了鱼填?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)毅戈。 經(jīng)常有香客問(wèn)我苹丸,道長(zhǎng),這世上最難降的妖魔是什么苇经? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任赘理,我火速辦了婚禮,結(jié)果婚禮上扇单,老公的妹妹穿的比我還像新娘商模。我一直安慰自己,他們只是感情好蜘澜,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布施流。 她就那樣靜靜地躺著,像睡著了一般鄙信。 火紅的嫁衣襯著肌膚如雪嫂沉。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,031評(píng)論 1 285
  • 那天扮碧,我揣著相機(jī)與錄音趟章,去河邊找鬼。 笑死慎王,一個(gè)胖子當(dāng)著我的面吹牛蚓土,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播赖淤,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼蜀漆,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了咱旱?” 一聲冷哼從身側(cè)響起确丢,我...
    開(kāi)封第一講書(shū)人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤绷耍,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后鲜侥,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體褂始,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年描函,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了崎苗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡舀寓,死狀恐怖胆数,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情互墓,我是刑警寧澤必尼,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站篡撵,受9級(jí)特大地震影響判莉,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜酸休,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望祷杈。 院中可真熱鬧斑司,春花似錦、人聲如沸但汞。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)私蕾。三九已至僵缺,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間踩叭,已是汗流浹背磕潮。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留容贝,地道東北人自脯。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像斤富,于是被迫代替她去往敵國(guó)和親膏潮。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 1 場(chǎng)景問(wèn)題# 1.1 報(bào)價(jià)管理## 向客戶(hù)報(bào)價(jià)满力,對(duì)于銷(xiāo)售部門(mén)的人來(lái)講焕参,這是一個(gè)非常重大轻纪、非常復(fù)雜的問(wèn)題,對(duì)不同的...
    七寸知架構(gòu)閱讀 5,052評(píng)論 9 62
  • 1 場(chǎng)景問(wèn)題 1.1 報(bào)價(jià)管理 向客戶(hù)報(bào)價(jià)叠纷,對(duì)于銷(xiāo)售部門(mén)的人來(lái)講刻帚,這是一個(gè)非常重大、非常復(fù)雜的問(wèn)題讲岁,對(duì)不同的客戶(hù)要...
    4e70992f13e7閱讀 3,071評(píng)論 2 16
  • 本文僅僅為入門(mén)我擂,高手勿噴。 實(shí)際工作中缓艳,我們總會(huì)遇到類(lèi)似如下的需求:某支付系統(tǒng)接入以下幾種商戶(hù)進(jìn)行充值:易寶網(wǎng)易校摩,...
    JarvanMo閱讀 14,229評(píng)論 14 26
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)阶淘,斷路器衙吩,智...
    卡卡羅2017閱讀 134,599評(píng)論 18 139
  • 1.問(wèn)題例1:一個(gè)菜單功能能夠根據(jù)用戶(hù)的“皮膚”首選項(xiàng)來(lái)決定是否采用水平的還是垂直的排列形式。同事可以靈活增加菜單...
    小飛豬閱讀 358評(píng)論 0 0