策略模式

作者: 杨千嬅染了红头发 | 来源:发表于2018-03-26 00:31 被阅读30次

策略模式: 要实现某一个功能,有多种方案可以选择。我们定义一系列的算法,把它们一个个封装起来,并且使它们可以相互替换。策略模式让算法独立于使用它们的客户端而独立变化。

基本理解

  • 当不同的行为堆砌在一个类中的时候,就很难避免使用条件语句来选择合适的行为。将这些行为封装在一个个独立的Strategy类中,可以再使用这些行为的类来消除条件语句。
  • 策略模式就是用来封装算法的, 但在实践中,我们发现可以用它来封装几乎任何类型的规则,只要在实际开发中遇到了需要根据不同的情况选择不同的业务规则,就可以考虑使用策略模式处理这种变化的可能性。
  • 在基本的策略模式中,选择所用具体实现的职责油客户端对象承受,并转给策略模式的 Context 对象。
  • 面向对象软件设计中,我们可以把相关算法分离为不用的类,成为策略。
  • 策略模式中的一个关键角色是策略类, 它为所有支持的或者相关的算法声明了一个共同接口。

结构图

策略模式(Strategy)结构图

责任链角色功能:

环境类(Context): 维护一个对 Strategy 对象的引用,可避免高层模块对策略的直接调用。
抽象策略类(Strategy) :定义所有支持的算法的公共接口。
具体策略类(ConcreteStrategy): 以 Strategy 抽象类定义的公共接口来实现具体的算法。
在策略模式中,调用算法的主体封装到了环境类Context中,抽象策略类(Strategy)一般是一个接口,目的只是为了定义接口的规范,里面一般不包含逻辑。

Demo

一个电商APP平时在特殊的日期会做一些活动,用户在最后结算订单的时候可以选择不同的优惠活动,如使用满200减30的优惠券,7折或者20无限制抵用券,我们就用这个例子写一个 demo。

PriceCalculationAbstract(价格计算的抽象类),对应抽象策略类(Strategy)
这个抽象类里面定义了一个关于价格计算的公共接口。

@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 满减优惠类 ,继承自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 环境类

@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

客户端调用

   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));

打印结果

最终价格 270

相关文章

网友评论

    本文标题:策略模式

    本文链接:https://www.haomeiwen.com/subject/ofcoqftx.html