《大話設(shè)計(jì)模式》第 2 章 - 策略模式 的 Swift 實(shí)現(xiàn)猪腕。
問(wèn)題
做一個(gè)商場(chǎng)收銀軟件,根據(jù)不同促銷方案返回不同的應(yīng)收款金額奢浑。
解決
策略模式定義了算法家族琼掠,分別封裝起來(lái)怕吴,讓它們之間可以互相替換,此模式讓算法的變化县踢,不會(huì)影響到使用算法的客戶械哟。
1. Strategy 類,定義算法的公共接口
Swift 中沒(méi)有抽象類殿雪,用一個(gè)協(xié)議來(lái)做聲明暇咆,聲明具體算法需要實(shí)現(xiàn)的方法。
//現(xiàn)金收費(fèi)協(xié)議
protocol CashSuper{
func acceptCash(money: Double) -> Double
}
2. ConcreteStrategy丙曙,封裝具體的算法爸业,繼承于 Strategy
實(shí)現(xiàn)具體算法的各個(gè)類遵循 CashSuper 協(xié)議,實(shí)現(xiàn)協(xié)議方法亏镰。
// MARK: 運(yùn)算類
//正常收費(fèi)
final class CashNormal: CashSuper{
func acceptCash(money: Double) -> Double {
return money
}
}
//打折收費(fèi)
final class CashRebate: CashSuper{
private var moneyRebate: Double = 1
init(moneyRebate: Double) {
self.moneyRebate = moneyRebate
}
func acceptCash(money: Double) -> Double {
return money * moneyRebate
}
}
//返利收費(fèi)
final class CashReturn: CashSuper{
private var moneyCondition: Double = 0.0
private var moneyReturn: Double = 0.0
init(moneyCondition: Double, moneyReturn: Double){
self.moneyCondition = moneyCondition
self.moneyReturn = moneyReturn
}
func acceptCash(money: Double) -> Double {
var result = money
if (money >= moneyCondition) {
result = money - floor(money / moneyCondition) * moneyReturn
}
return result
}
}
3. Context扯旷,傳入 ConcreteStrategy,維護(hù)對(duì) Strategy 的引用索抓。
策略模式結(jié)合簡(jiǎn)單工廠钧忽,工廠根據(jù)傳入的字符串生成對(duì)應(yīng)的具體算法,實(shí)例化具體算法的過(guò)程由客戶端轉(zhuǎn)移到 context 類逼肯。然后通過(guò)調(diào)用 GetResult 方法得到計(jì)算結(jié)果耸黑,讓具體算法與客戶端隔離。
//策略模式結(jié)合簡(jiǎn)單工廠
final class CashContext{
private var cs: CashSuper?
init(type: String){
switch type {
case "正常收費(fèi)":
self.cs = CashNormal()
case "滿300返100":
self.cs = CashReturn(moneyCondition: 300,moneyReturn: 100)
case "打8折":
self.cs = CashRebate(moneyRebate: 0.8)
default:
break
}
}
func GetResult(money: Double) -> Double{
guard let cs = cs else {
return money
}
return cs.acceptCash(money: money)
}
}
第一章簡(jiǎn)單工廠模式中篮幢,我們需要讓客戶端認(rèn)識(shí)兩個(gè)類大刊,CashSuper 和 CashFactory,而策略模式與簡(jiǎn)單工廠模式結(jié)合的用法三椿,客戶端之需要認(rèn)識(shí)一個(gè)類 CashContext 就可以了缺菌,連算法的父類(協(xié)議) CashSuper 都不讓客戶端認(rèn)識(shí)。耦合度更低搜锰。
測(cè)試
let result1 = CashContext(type:"正常收費(fèi)").GetResult(money: 580)
let result2 = CashContext(type:"滿300返100").GetResult(money: 580)
let result3 = CashContext(type:"打8折").GetResult(money: 580)
print("總價(jià):580伴郁,正常收費(fèi) = \(result1); 滿300返100 = \(result2)蛋叼;打8折 = \(result3)")
總結(jié)
用相同的方法調(diào)用任何一個(gè)算法焊傅,減少各種算法類與使用算法類(客戶端)之間的耦合。
當(dāng)不同行為堆砌在一個(gè)類中時(shí)鸦列,很難避免使用條件語(yǔ)句來(lái)選擇合適的行為租冠。將這些行為封裝在一個(gè)個(gè)獨(dú)立的 Strategy 類中,就可以在使用行為的類中消除條件語(yǔ)句薯嗤。
策略模式封裝了變化。策略模式就是用來(lái)封裝算法的纤泵,在實(shí)踐中骆姐,它可以用來(lái)封裝任何類型的規(guī)則镜粤,如果需求需要在不同時(shí)間應(yīng)用不同的業(yè)務(wù)規(guī)則,那么就可以考慮用策略模式來(lái)處理這種變化的可能性玻褪。