策略模式: 要實現(xiàn)某一個功能,有多種方案可以選擇希坚。我們定義一系列的算法,把它們一個個封裝起來,并且使它們可以相互替換。策略模式讓算法獨立于使用它們的客戶端而獨立變化玛痊。
基本理解
- 當不同的行為堆砌在一個類中的時候,就很難避免使用條件語句來選擇合適的行為锌俱。將這些行為封裝在一個個獨立的Strategy類中,可以再使用這些行為的類來消除條件語句迎膜。
- 策略模式就是用來封裝算法的, 但在實踐中,我們發(fā)現(xiàn)可以用它來封裝幾乎任何類型的規(guī)則,只要在實際開發(fā)中遇到了需要根據(jù)不同的情況選擇不同的業(yè)務規(guī)則,就可以考慮使用策略模式處理這種變化的可能性。
- 在基本的策略模式中,選擇所用具體實現(xiàn)的職責油客戶端對象承受,并轉(zhuǎn)給策略模式的 Context 對象神妹。
- 面向?qū)ο筌浖O(shè)計中,我們可以把相關(guān)算法分離為不用的類,成為策略颓哮。
- 策略模式中的一個關(guān)鍵角色是策略類, 它為所有支持的或者相關(guān)的算法聲明了一個共同接口。
結(jié)構(gòu)圖
責任鏈角色功能:
環(huán)境類(Context): 維護一個對 Strategy 對象的引用,可避免高層模塊對策略的直接調(diào)用鸵荠。
抽象策略類(Strategy) :定義所有支持的算法的公共接口冕茅。
具體策略類(ConcreteStrategy): 以 Strategy 抽象類定義的公共接口來實現(xiàn)具體的算法。
在策略模式中,調(diào)用算法的主體封裝到了環(huán)境類Context中,抽象策略類(Strategy)一般是一個接口,目的只是為了定義接口的規(guī)范,里面一般不包含邏輯蛹找。
Demo
一個電商APP平時在特殊的日期會做一些活動,用戶在最后結(jié)算訂單的時候可以選擇不同的優(yōu)惠活動,如使用滿200減30的優(yōu)惠券,7折或者20無限制抵用券,我們就用這個例子寫一個 demo姨伤。
PriceCalculationAbstract(價格計算的抽象類),對應抽象策略類(Strategy)
這個抽象類里面定義了一個關(guān)于價格計算的公共接口。
@interface PriceCalculationAbstract : NSObject
- (double)calculationPrice:(double)price;
@end
@implementation PriceCalculationAbstract
- (double)calculationPrice:(double)price
{
return 0;
}
@end
PriceDiscount折扣計算方法類,繼承自PriceCalculationAbstract類,對應的是具體策略類
@interface PriceDiscount : PriceCalculationAbstract
- (instancetype)initWithPriceDiscount:(float)priceDiscount;
@end
@interface PriceDiscount()
@property (nonatomic, assign) float priceDiscount;
@end
@implementation PriceDiscount
- (instancetype)initWithPriceDiscount:(float)priceDiscount
{
self = [super init];
if (self) {
_priceDiscount = priceDiscount;
}
return self;
}
- (double)calculationPrice:(double)price
{
return self.priceDiscount *price;
}
@end
PriceRelief 滿減優(yōu)惠類 ,繼承自PriceCalculationAbstract類,對應的是具體策略類
@interface PriceRelief : PriceCalculationAbstract
- (instancetype)initWithConditionValue:(double)conditionValue
relief:(double)relief;
@end
@interface PriceRelief()
@property (nonatomic, assign) double conditionValue;
@property (nonatomic, assign) double relief;
@end
@implementation PriceRelief
- (instancetype)initWithConditionValue:(double)conditionValue
relief:(double)relief
{
self = [super init];
if (self) {
_conditionValue = conditionValue;
_relief = relief;
}
return self;
}
- (double)calculationPrice:(double)price
{
if (price >= self.conditionValue) {
return price - self.relief;
}
return price;
}
@end
DirectRelief 直接減免金額方法類, ,繼承自PriceCalculationAbstract類,對應的是具體策略類
@interface DirectRelief : PriceCalculationAbstract
- (instancetype)initWithRelief:(double)relief;
@end
@interface DirectRelief()
@property (nonatomic ,assign) double relief;
@end
@implementation DirectRelief
- (instancetype)initWithRelief:(double)relief
{
self = [super init];
if (self) {
_relief = relief;
}
return self;
}
- (double)calculationPrice:(double)price
{
return price - self.relief;
}
@end
PriceContext 價格生成類,對應 Context 環(huán)境類
@class PriceCalculationAbstract;
@interface PriceContext : NSObject
@property (nonatomic, strong) PriceCalculationAbstract *priceCalculationAbstract;
- (double)getResult:(double)price;
@end
@implementation PriceContext
- (double)getResult:(double)price
{
double result = price;
if (self.priceCalculationAbstract) {
result = [self.priceCalculationAbstract calculationPrice:price];
}
return result;
}
@end
客戶端調(diào)用
PriceContext *priceContext = [[PriceContext alloc] init];
PriceCalculationAbstract *priceCalculationAbstract = [[PriceCalculationAbstract alloc] init];
int type = 2;
switch (type) {
case 1:
priceCalculationAbstract = [[PriceDiscount alloc] initWithPriceDiscount:0.7];
break;
case 2:
priceCalculationAbstract = [[PriceRelief alloc] initWithConditionValue:200 relief:30];
break;
case 3:
priceCalculationAbstract = [[DirectRelief alloc] initWithRelief:20];
break;
default:
break;
}
priceContext.priceCalculationAbstract = priceCalculationAbstract;
int result = [priceContext getResult:300];
NSLog(@"最終價格 %@", @(result));
打印結(jié)果
最終價格 270