策略模式

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

基本理解

  • 当不同的行为堆砌在一个类中的时候,就很难避免使用条件语句来选择合适的行为。将这些行为封装在一个个独立的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
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容

  • 1 场景问题# 1.1 报价管理## 向客户报价,对于销售部门的人来讲,这是一个非常重大、非常复杂的问题,对不同的...
    七寸知架构阅读 5,054评论 9 62
  • 1 场景问题 1.1 报价管理 向客户报价,对于销售部门的人来讲,这是一个非常重大、非常复杂的问题,对不同的客户要...
    4e70992f13e7阅读 3,071评论 2 16
  • 本文仅仅为入门,高手勿喷。 实际工作中,我们总会遇到类似如下的需求:某支付系统接入以下几种商户进行充值:易宝网易,...
    JarvanMo阅读 14,229评论 14 26
  • 目录 本文的结构如下: 引言 什么是策略模式 模式的结构 典型代码 代码示例 策略模式和模板方法模式的区别 优点和...
    w1992wishes阅读 843评论 1 7
  • 1.概述 demo地址,欢迎star 例子1: 商场产品销售:可以正常价格售出,可以打折形式售出,也可以通过积分的...
    barry阅读 740评论 0 1