iOS设计模式六(模板,策略,命令)

承接上文iOS设计模式五(访问者,装饰,责任链)
本文为算法封装--获取源码

目录
1 模板模式
2 策略模式
3 命令模式


1 模板模式

模板模式顾名思义,套用模板.应该是最简单常用的模式了吧,就是俩字:复用!
继承也是复用的一种方式,在抽象类或者父类中定义公共的方法,在子类中重写或重载实现

稍微看个继承的代码😂
做一顿饭有很多选择,中餐,西餐.
父类OSZFood.h

#import <Foundation/Foundation.h>
@interface OSZFood : NSObject
//做饭
- (void)make;
//米饭
- (void)makeMeal;
//肉
- (void)makeMeat;
//蔬菜
- (void)makeVegetables;
//预留的空方法,子类去添加 所谓的"钩子"
- (void)makeOther;
@end

OSZFood.m

#import "OSZFood.h"
@implementation OSZFood
- (void)make{
    [self makeMeal];
    [self makeMeat];
    [self makeVegetables];
}
- (void)makeMeal{
    NSLog(@"做主食");
}
- (void)makeMeat{
    NSLog(@"做肉");
}
- (void)makeVegetables{
    NSLog(@"做菜");
}
- (void)makeOther{  
}
@end

中餐OSZChineseFood.h

#import "OSZFood.h"
@interface OSZChineseFood : OSZFood

//重写做菜方法,加入中餐特有的食物
- (void)makeVegetables;
- (void)chao;
@end

OSZChineseFood.m

#import "OSZChineseFood.h"
@implementation OSZChineseFood

//重写make方法,加入中餐特有的食物
- (void)makeVegetables{
    [self chao];
}
- (void)chao{
    NSLog(@"炒个菜");
}
@end

西餐 OSZWesternFood.h

#import "OSZFood.h"
@interface OSZWesternFood : OSZFood

//重写做肉方法,加入西餐特有的牛排
- (void)makeMeat;
- (void)steak;
@end

OSZWesternFood.m

#import "OSZWesternFood.h"
@implementation OSZWesternFood

- (void)makeMeat{
    [self steak];
}
- (void)steak{
    NSLog(@"煎个牛排");
}
@end

控制器OSZThirteenVC.m

#import "OSZThirteenVC.h"
#import "OSZChineseFood.h"
#import "OSZWesternFood.h"
@interface OSZThirteenVC ()
@end
@implementation OSZThirteenVC
- (void)viewDidLoad {
    [super viewDidLoad];
    OSZChineseFood *c = [[OSZChineseFood alloc]init];
    [c make];    //做主食  //做肉 //炒个菜

    OSZWesternFood *w = [[OSZWesternFood alloc]init];
    [w make];    //做主食  //煎个牛排  //做菜
}
@end

没什么可看的,就是个继承实现的简单复用,是抽出共同行为放入框架类中的基本手段


2 策略模式

策略模式的类结构

策略模式中一个关键角色是策略类strategy,它为所有支持的或相关的算法声明了一个共同接口,下面三个是具体策略类concreteStrategy,分别根据context的需求实现不同的效果,
调用时context使用strategy声明的方法,根据枚举值选用具体策略类concreteStrategy就可以了

看到这里是不是感觉很简单,其实我们在写app做一个界面的时候就自然的用到了,
context就是视图View,策略类就是控制器Controller,
我们复用同一个View,Controller调用不同的接口,或者干脆换个Controller,
在这个View上显示不同的控件(比如Label,按钮Button,或者自定义的控件),或者显示不同的数据Model

简单的MVC中策略模式例子:
做一个简单的视图,根据控制器不同显示不同的数据
OSZFourteenView.h

#import <UIKit/UIKit.h>
@interface OSZFourteenView : UIView
@property (nonatomic, weak) UILabel *lbl;
@property (nonatomic, weak) UIButton *btn;
@end

OSZFourteenView.m

#import "OSZFourteenView.h"
@implementation OSZFourteenView
- (instancetype)initWithFrame:(CGRect)frame{
    self = [super initWithFrame:frame];
    if (self){
        [self setupUI];
    }
    return self;
}

- (void)setupUI{
    self.backgroundColor = [UIColor whiteColor];
    
    CGRect rect = CGRectMake(100,100,150,30);
    UILabel *titleLbl = [[UILabel alloc]initWithFrame:rect];
    titleLbl.text = @"";
    titleLbl.textColor = [UIColor blackColor];
    titleLbl.font = [UIFont boldSystemFontOfSize:22];
    titleLbl.textAlignment = NSTextAlignmentCenter;
    [self addSubview:titleLbl];
    self.lbl = titleLbl;
    
    rect = CGRectMake(100,200,200,30);
    UIButton *button = [[UIButton alloc]initWithFrame:rect];
    [button setTitle:@"" forState:UIControlStateNormal];
    [button setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
    button.titleLabel.font = [UIFont systemFontOfSize:22];
    [self addSubview:button];
    self.btn = button;
}
@end

两个控制器:
OSZFourteenVC.m

#import "OSZFourteenVC.h"
#import "OSZFourteenView.h"
#import "OSZFourteenVC2.h"
@interface OSZFourteenVC ()
@end

@implementation OSZFourteenVC
- (void)viewDidLoad {
    [super viewDidLoad];
    OSZFourteenView *view14 = [[OSZFourteenView alloc]init];
    self.view = view14;
    view14.lbl.text = @"第一个控制器";
    [view14.btn setTitle:@"跳到第二个控制器" forState:UIControlStateNormal];
    [view14.btn addTarget:self action:@selector(turn) forControlEvents:UIControlEventTouchUpInside];
}

- (void)turn{
    [self presentViewController:[[OSZFourteenVC2 alloc]init] animated:YES completion:nil];
}
@end

OSZFourteenVC2.m

#import "OSZFourteenVC2.h"
#import "OSZFourteenView.h"
#import "OSZFourteenVC.h"
@interface OSZFourteenVC2 ()
@end

@implementation OSZFourteenVC2

- (void)viewDidLoad {
    [super viewDidLoad];
    OSZFourteenView *view14 = [[OSZFourteenView alloc]init];
    self.view = view14;
    view14.lbl.text = @"第二个控制器";
    [view14.btn setTitle:@"跳到第一个控制器" forState:UIControlStateNormal];
    [view14.btn addTarget:self action:@selector(turn) forControlEvents:UIControlEventTouchUpInside];
}

- (void)turn{
    [self dismissViewControllerAnimated:YES completion:nil];
}
@end

策略模式与装饰模式看起来有点像,但是装饰器是在对象外面叠加,而策略模式是在内部更换内容


3 命令模式

把指令封装在命令对象中,命令对象可以被传递并且被不同的客户端复用,客户端可以把它参数化并置入队列或日志中
命令模式消除了作为对象的动作和执行它的接收器之间的绑定.

看完了概念是不是感觉似曾相识?
实际上命令模式是Cocoa Touch框架收录的模式之一
NSInvocation与NSUndoManager和"目标-动作"机制是框架中对这个模式的典型应用

NSInvocation对象封装运行时库以向接收器转发执行消息所需的所有必要信息,是调用方法的一种方式
NSUndoManager作为通用的撤销栈的管理类,能够逆转,撤销操作
"目标-动作"机制就是 目标(Target)与动作(Action),即用performSelector调用方法的一种方式

一个用NSUndoManager实现的支持撤销功能的画板:

定义线OSZLine.h

#import <Foundation/Foundation.h>
@interface OSZLine : NSObject

@property (nonatomic,assign) CGPoint begin;
@property (nonatomic,assign) CGPoint end;
@property (nonatomic,strong) UIColor *color;

@end

OSZLine.m

#import "OSZLine.h"
@implementation OSZLine
-(instancetype)init{
    self = [super init];
    if (self) {
        [self setColor:[UIColor blackColor]];
    }
    return self;
}
@end

画板OSZDrawView.h

#import <UIKit/UIKit.h>
#import "OSZLine.h"
@interface OSZDrawView : UIView

@property (nonatomic) OSZLine *currentLine;
@property (nonatomic) NSMutableArray *linesCompleted;
//回退
@property (nonatomic, weak) UIButton *undoBtn;
//前进
@property (nonatomic, weak) UIButton *redoBtn;

@end

OSZDrawView.m

#import "OSZDrawView.h"

@implementation OSZDrawView

- (instancetype)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self)
    {
        [self setupUI];
    }
    return self;
}

- (void)setupUI
{
    self.backgroundColor = [UIColor whiteColor];
    self.linesCompleted = [[NSMutableArray alloc] init];
    [self setMultipleTouchEnabled:YES];
    
    [self becomeFirstResponder];
    
    CGFloat origin = 40;
    CGFloat W = (kScreenWidth - 4 * origin ) * 0.5;
    CGFloat H = 44;
    CGFloat X = origin;
    CGFloat Y = kScreenHeight - origin - H;
    CGRect rect = CGRectMake(X,Y,W,H);
    UIButton *undoBtn = [[UIButton alloc]initWithFrame:rect];
    [undoBtn setTitle:@"回退" forState:UIControlStateNormal];
    [undoBtn setBackgroundColor:[UIColor grayColor]];
    [undoBtn setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
    undoBtn.titleLabel.font = [UIFont systemFontOfSize:17];
    undoBtn.layer.cornerRadius = 5;
    undoBtn.layer.masksToBounds = YES;
    self.undoBtn = undoBtn;
    [self addSubview:undoBtn];
    [self.undoBtn addTarget:self action:@selector(undo) forControlEvents:UIControlEventTouchUpInside];
    
    X = 3 * origin + W;
    rect = CGRectMake(X,Y,W,H);
    UIButton *redoBtn = [[UIButton alloc]initWithFrame:rect];
    [redoBtn setTitle:@"前进" forState:UIControlStateNormal];
    [redoBtn setBackgroundColor:[UIColor grayColor]];
    [redoBtn setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
    redoBtn.titleLabel.font = [UIFont systemFontOfSize:17];
    redoBtn.layer.cornerRadius = 5;
    redoBtn.layer.masksToBounds = YES;
    self.undoBtn = redoBtn;
    [self addSubview:redoBtn];
    [self.undoBtn addTarget:self action:@selector(redo) forControlEvents:UIControlEventTouchUpInside];
}


// Only override drawRect: if you perform custom drawing.
// An empty implementation adversely affects performance during animation.
- (void)drawRect:(CGRect)rect {
    CGContextRef context = UIGraphicsGetCurrentContext();
    
    CGContextSetLineWidth(context, 5.0);
    CGContextSetLineCap(context, kCGLineCapRound);
    for (OSZLine *line in self.linesCompleted) {
        [[line color] set];
        CGContextMoveToPoint(context, [line begin].x, [line begin].y);
        CGContextAddLineToPoint(context, [line end].x, [line end].y);
        CGContextStrokePath(context);
    }
}

//回退
- (void)undo
{
    if ([self.undoManager canUndo]) {
        [self.undoManager undo];
    }
}

//前进
- (void)redo
{
    if ([self.undoManager canRedo]) {
        [self.undoManager redo];
    }
}

//NSUndoManager 的实现原理是它作为一个记录器,每次数据变化,我们要用这个记录器记录一个相反的操作,
//当需要undo的时候,它通过执行这个相反的操作就可以实现了。
- (void)addLine:(OSZLine*)line
{
    [[self.undoManager prepareWithInvocationTarget:self] removeLine:line];
    [self.linesCompleted addObject:line];
    [self setNeedsDisplay];
}

- (void)removeLine:(OSZLine*)line
{
    if ([self.linesCompleted containsObject:line]) {
        [[self.undoManager prepareWithInvocationTarget:self] addLine:line];
        [self.linesCompleted removeObject:line];
        [self setNeedsDisplay];
    }
}

- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
{
    for (UITouch *t in touches)
    {
        CGPoint loc = [t locationInView:self];
        self.currentLine.end = loc;
        
        if (self.currentLine)
        {
            [self addLine:self.currentLine];
        }
        OSZLine *newLine = [[OSZLine alloc] init];
        newLine.begin = loc;
        newLine.end = loc;
        self.currentLine = newLine;
    }
}



- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
    //NSUndoManager有一个分组的概念,就是为了解决这类问题的
    //将所有的笔画放在一个组中
    [self.undoManager beginUndoGrouping];
    for (UITouch *t in touches) {
        // Create a line for the value
        CGPoint loc = [t locationInView:self];
        OSZLine *newLine = [[OSZLine alloc] init];
        newLine.begin = loc;
        newLine.end = loc;
        self.currentLine = newLine;
    }
}

- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
{
    [self setNeedsDisplay];
    [self.undoManager endUndoGrouping];
}

@end

控制器OSZFifteenVC.m

#import "OSZFifteenVC.h"
#import "OSZDrawView.h"

@interface OSZFifteenVC ()

@property (nonatomic, weak) OSZDrawView *drawView;

@end

@implementation OSZFifteenVC

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    
    //NSUndoManager应用
    OSZDrawView *drawView = [[OSZDrawView alloc]initWithFrame:self.view.frame];
    self.drawView = drawView;
    [self.view addSubview:drawView];
    
    //NSInvocation应用
    [self invocationMethod];
    
    //所谓的"目标-动作"机制就是 目标(Target)与动作(Action)
    [self performSelector:@selector(invocationMethod)];
}

//命令模式是Cocoa Touch框架收录的模式之一
//NSInvocation与NSUndoManager和"目标-动作"机制是框架中对这个模式的典型应用
- (void)invocationMethod
{
    //NSInvocation;用来包装方法和对应的对象,它可以存储方法的名称,对应的对象,对应的参数,
    /*
     NSMethodSignature:签名:再创建NSMethodSignature的时候,必须传递一个签名对象,签名对象的作用:用于获取参数的个数和方法的返回值
     */
    //创建签名对象的时候不是使用NSMethodSignature这个类创建,而是方法属于谁就用谁来创建
    NSMethodSignature*signature = [OSZFifteenVC instanceMethodSignatureForSelector:@selector(sendMessageWithNumber:WithContent:)];
    //1、创建NSInvocation对象
    NSInvocation*invocation = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = self;
    //invocation中的方法必须和签名中的方法一致。
    invocation.selector = @selector(sendMessageWithNumber:WithContent:);
    /*第一个参数:需要给指定方法传递的值
     第一个参数需要接收一个指针,也就是传递值的时候需要传递地址*/
    //第二个参数:需要给指定方法的第几个参数传值
    NSString*number = @"13812345678";
    //注意:设置参数的索引时不能从0开始,因为0已经被self占用,1已经被_cmd占用
    [invocation setArgument:&number atIndex:2];
    NSString*number2 = @"哈哈哈";
    [invocation setArgument:&number2 atIndex:3];
    //2、调用NSInvocation对象的invoke方法
    //只要调用invocation的invoke方法,就代表需要执行NSInvocation对象中制定对象的指定方法,并且传递指定的参数
    [invocation invoke];
}

- (void)sendMessageWithNumber:(NSString*)number WithContent:(NSString*)content{
    NSLog(@"电话号%@,姓名%@",number,content);
}
@end

扩展:
iOS: 为画板App增加 Undo/Redo(撤销/重做)操作

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

推荐阅读更多精彩内容

  • 设计模式汇总 一、基础知识 1. 设计模式概述 定义:设计模式(Design Pattern)是一套被反复使用、多...
    MinoyJet阅读 3,903评论 1 15
  • 设计模式基本原则 开放-封闭原则(OCP),是说软件实体(类、模块、函数等等)应该可以拓展,但是不可修改。开-闭原...
    西山薄凉阅读 3,753评论 3 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,598评论 18 139
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,019评论 4 62
  • 铁竹轩这家伙居然会开车。 我是怀着忐忑不安的心情坐上车的,没有他开的还挺好的。 “你什么时候学的?” “沐导教的。...
    沐弘晨阅读 332评论 0 22