模版方法模式(Template Method Pattern)

模版方法模式:在一个方法中定义算法的骨架。而将一些具体的步骤延续到子类中。模版方法模式让子类在不改变算法逻辑的基础上可以重新定义算法的某些步骤。

模版方法模式简称为模版模式是行为性模式之一。看定义我们大概可以知道的是,模版方法模式提供一个算法的骨架,也就是模版方法。而算法的某些关键性的步骤由子类具体实现。以此来提高系统的灵活性。

模版方法模式

模版方法的强大之处在于其一通过模版方法,将部分关键性的算法放到子类来实现,提高系统的灵活性,其二是可以很方便的提供钩子。
钩子是声明在抽象类中的方法,但只有空的/默认的实现。钩子的存在可以让子类有能力对算法的不同点进行挂钩。

举个栗子
比如说我们开了一家饮料店。
整体上讲茶,咖啡,还有其他的饮料的做法都是相似的。烧开水,浸泡饮料的响应成分(茶叶,咖啡,可可粉,奶茶粉等等),将饮料倒入杯中,加配料。

烧开水,将饮料倒入杯中是固定的写法,没有其他可操作的成分。
浸泡饮料涉及到具体的饮品。所以需要在子类中实现。
加配料是可选的。有的顾客想要加,有的顾客不想加。
所以钩子可以加在配料哪里。

如果实现了钩子,说明是需要加配料的,如果没有实现钩子,说明不需要加配料。

饮料抽象类 这里preapreRecipe 就是模版方法。isAddingComditions方法是钩子方法。

#import "CaffineBaverate.h"

@implementation CaffineBaverate
//模版方法
-(void)preapreRecipe {

    [self boilWater];
    [self brew];
    [self pureInCup];
    if ([self isAddingComditions]) {
       [self addComditions];
    }
}
     
-(void)boilWater {
    
    NSLog(@"烧开水");
}

-(void)pureInCup {
    
    NSLog(@"放入杯中");
}

- (void)brew {
    
    @throw [NSException exceptionWithName:@"抽象类" reason:@"抽象类中的抽象方法由子类实现,不能直接调用抽象类" userInfo:nil];
}

-(void)addComditions {
    
    @throw [NSException exceptionWithName:@"抽象类" reason:@"抽象类中的抽象方法由子类实现,不能直接调用抽象类" userInfo:nil];
}

-(BOOL)isAddingComditions {
    return YES;
}

@end

具体饮料茶(继承于抽象饮料)

#import "Tea.h"

@interface Tea ()

@property (nonatomic ,strong)NSString * text;

@end

@implementation Tea

-(void)brew {
    NSLog(@"浸泡茶叶");
}

-(void)addComditions {
    NSLog(@"给茶添加冰和冰糖");
}

//实现了钩子
- (BOOL)isAddingComditions {
    
    char cString[10];
    
    printf("需要添加点配料吗(输入y/n):");
    
    scanf("%s",cString);
    
    if ([[NSString stringWithCString:cString encoding:NSUTF8StringEncoding] containsString:@"y"]) {
        return true;
    }
    
    return false;
}

@end

具体的使用

Tea * tea = [[Tea alloc] init];
//preapreRecipe是抽象类中的算法骨架,该算法中的brew和addComditions延迟到Tea中实现。
[tea preapreRecipe];

模版方法模式也有其他形式的变体。

举个栗子

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@protocol Comparable <NSObject>

-(BOOL)compareTo:(id<Comparable>)obj;

@end

NS_ASSUME_NONNULL_END

这是一个数组的排序类(这里将compareTo方法在子类实现)

#import "NSObject+sort.h"

@implementation NSObject (sort)

+(NSMutableArray *)sortWithArray:(NSMutableArray<Comparable> *)sortArray {
    sortArray = [sortArray mutableCopy];
    for (int i = 0; i<= sortArray.count - 1; i++) {
        for (int j = i ; j >= 1  ; j --) {
            if ([sortArray[j - 1] compareTo:sortArray[j]]) {
                [sortArray exchangeObjectAtIndex:j withObjectAtIndex:j-1];
            }
        }
    }
    return sortArray;
}
@end

这是具体类

#import "IDCard.h"

@implementation IDCard

-(BOOL)compareTo:(IDCard *)obj {
    
    if (self.idString.length != 18) exit(0) ;
    if (obj.idString.length != 18) exit(0) ;
    int year = [[self.idString substringWithRange:NSMakeRange(6, 4)] intValue];
    int comYear = [[obj.idString substringWithRange:NSMakeRange(6, 4)] intValue];
    if (year < comYear) {
        return YES;
    }else if (year == comYear)
    {
        int month = [[self.idString substringWithRange:NSMakeRange(10, 2)] intValue];
        int comMonth = [[obj.idString substringWithRange:NSMakeRange(10, 2)] intValue];
        if (month < comMonth) {
            return YES;
        }else if (month == comMonth) {
            int day = [[self.idString substringWithRange:NSMakeRange(12, 2)] intValue];
            int comDay = [[obj.idString substringWithRange:NSMakeRange(12, 2)] intValue];
            if (day < comDay) {
                return YES;
            }else {
                return NO;
            }
        }
    }
    return NO;
}
-(NSString *)description {
    return self.idString;
}

@end

这个模式意在提供一个算法骨架,而子类去实现其中的某些步骤,而这里的排序却并非如此。但是这里的实现却符合模版方法模式的精神,亦算作模版方法模式。

优点

提升了代码的灵活性
增强了复用性
能很好的添加钩子,监听或者管理方法的调用

缺点

算法骨架不容易修改,违背开闭原则。

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

推荐阅读更多精彩内容