二十四、 命令模式

1. 何为模板方法模式

在软件系统中,“行为请求者”与“行为实现者”通常呈现一种“紧耦合”,但在某些场合,比如要对行为进行“记录、撤销、事务”等处理,这种无法抵御变化的耦合是不合适的。在这种情况下,将一组行为抽象为对象,实现二者之间的松耦合,这就是命令模式(Command Pattern)。命令模式的UML图见图1-1:

图1-1 命令模式

下面通过改变一个视图的明暗程度来体会命令模式的优缺点。

2. 非命令模式

(1)首先在ViewController添加三个按钮,并设置好相关属性和监听点击事件:

typedef enum : NSUInteger {
    hAddButtonTag = 0x11,
    hDelButtonTag,
    hRolButtonTag,
} ViewControllerEnumValue;

@interface ViewController ()
/** 接受者,执行任务者 */
@property (nonatomic,strong)Receiver *receiver;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 画出三个按钮
    // 调亮按钮 +
    UIButton* addBtn = [self addButtonWithTitle:@"+"
                                      withFrame:CGRectMake(30, 30, 40, 40)
                                     withAction:@selector(buttonsEvent:)
                                        withTag:hAddButtonTag];
    [self.view addSubview:addBtn];
    // 调暗按钮 -
    UIButton* delBtn = [self addButtonWithTitle:@"-"
                                      withFrame:CGRectMake(100, 30, 40, 40)
                                     withAction:@selector(buttonsEvent:)
                                        withTag:hDelButtonTag];
    [self.view addSubview:delBtn];
    // 撤销操作按钮
    UIButton* rolBtn = [self addButtonWithTitle:@"RoolBack"
                                      withFrame:CGRectMake(170, 30, 100, 40)
                                     withAction:@selector(buttonsEvent:)
                                        withTag:hRolButtonTag];
    [self.view addSubview:rolBtn];
    
    self.receiver = [[Receiver alloc] init];
    [self.receiver setClientView:self.view];
}


-(void)buttonsEvent:(UIButton*)btn{
    if (btn.tag == hAddButtonTag) {
        
        [self.receiver makeViewLighter:0.1f];
        
    }else if (btn.tag == hDelButtonTag){
        
        [self.receiver makeViewDarker:0.1f];
        
    }else if (btn.tag == hRolButtonTag){
        
    }
}

#pragma mark - 添加同类按钮的方法
// 增加相同按钮的方法相同,所以抽离出来
-(UIButton*)addButtonWithTitle:(NSString*)title withFrame:(CGRect)frame withAction:(SEL)sel withTag:(ViewControllerEnumValue)tag{
    UIButton* btn = [[UIButton alloc] initWithFrame:frame];
    [btn setTitle:title forState:UIControlStateNormal];
    [btn setTitleColor:[UIColor redColor] forState:UIControlStateNormal];
    [btn setTitle:@"🐶" forState:UIControlStateHighlighted];
    btn.layer.borderWidth = 1.0f;
    [btn addTarget:self action:sel forControlEvents:UIControlEventTouchUpInside];
    [btn setTag:tag];
    return btn;
}
@end

(2)然后创建任务执行者对象Receiver类,创建的实例对象负责执行调整View背景明亮暗度。

// Receiver任务执行者,有服务的对象,那么也有操作服务对象的具体行为

// 这里根据业务逻辑任务就是改变client的明亮程度

@interface Receiver : NSObject

/** 服务的对象 */
@property (nonatomic,strong)UIView *clientView;

// 增加亮度的行为
-(void)makeViewLighter:(CGFloat)quantity;
// 降低亮度的行为
-(void)makeViewDarker:(CGFloat)quantity;

@end
@interface Receiver ()
{
    CGFloat _hud;           // 色调
    CGFloat _saturation;    // 饱和度
    CGFloat _brightness;    // 亮度
    CGFloat _alpha;         // 透明度 alpha=1表示完全不透明
}
@end

@implementation Receiver

// 需要重写clientView的set方法,因为Receiver最开始要做的就是set获取UIView对象
// 在set对象的同时要获取当前client的状态,获取当前状态需要变量存储状态值
-(void)setClientView:(UIView *)clientView
{
    _clientView = clientView;
    
    UIColor *color = clientView.backgroundColor;
    [color getHue:&_hud saturation:&_saturation brightness:&_brightness alpha:&_alpha];
}

-(void)makeViewLighter:(CGFloat)quantity{
    _brightness += quantity;
    self.clientView.backgroundColor = [[UIColor alloc] initWithHue:_hud
                                                        saturation:_saturation
                                                        brightness:_brightness
                                                             alpha:_alpha];
}

-(void)makeViewDarker:(CGFloat)quantity{
    _brightness -= quantity;
    self.clientView.backgroundColor = [[UIColor alloc] initWithHue:_hud
                                                        saturation:_saturation
                                                        brightness:_brightness
                                                             alpha:_alpha];
    
}

@end

效果如下:

效果.gif

其实到这里,Receive对象相当于ViewController的代理,代理完成控制属于Viewcontroller管理和控制的View 的背景明亮暗度调整。只不过这个代理不是那么抽象而且也不是遵循了某个协议的。是具体而直接完成逻辑业务的。这个简单的模式,在项目或者不考虑拓展性或者 某个模块功能固定了,可以使用这个模式。

但是有的业务需求会需要记录存储和取出执行任务的或者信息传递命令的状态,像这里如果要添加撤销操作,就需要记录之前的操作,然后根据之前的操作回 退反过来操作,这时候,命令模式就能比较方便的实现这个逻辑了。命令模式其实就是将发送的命令信息抽象成一个类,然后具体信息就创建具体的类,并通过回调 者添加并执行命令对象的执行命令涉及到的任务方法,同时存储记录这个命令,那么这时候因为每次操作都能存储下来,所以再来设计撤销操作就很容易了。

3. 命令模式

(1)需要遵守的协议

@protocol InvokerProtocol <NSObject>
/*
 * 这个协议是Invoker调用者要求的协议,希望遵循这个协议的类,实现了必须要实现的两个方法
 * 因为这两个方法,在Invoker中一定会被调用
 */
@required

/**
 *  命令的执行
 */
- (void)excute;

/**
 *  撤销命令
 */
- (void)rollBackExcute;

@end

(2)变亮和变暗的命令

@interface LighterCommand : NSObject <InvokerProtocol>

- (instancetype)initWithReceiver:(Receiver*)receiver withParamter:(CGFloat)paramter;

@end


@interface LighterCommand ()
/** 接受者 */
@property (nonatomic, strong) Receiver *receiver;
/** 数值 */
@property (nonatomic, assign) CGFloat paramter;
@end

@implementation LighterCommand


- (instancetype)initWithReceiver:(Receiver*)receiver withParamter:(CGFloat)paramter
{
    self = [super init];
    if (self) {
        self.receiver = receiver;
        self.paramter = paramter;
    }
    return self;
}

/**
 *  命令的执行 思考一下,命令怎么执行让任务实现?
 *  
 */
- (void)excute{
    [self.receiver makeViewLighter:self.paramter];
}

/**
 *  撤销命令
 */
- (void)rollBackExcute{
    [self.receiver makeViewDarker:self.paramter];
}

@end
@interface DarkerCommand : NSObject  <InvokerProtocol>

- (instancetype)initWithReceiver:(Receiver*)receiver withParamter:(CGFloat)paramter;

@end


@interface DarkerCommand ()
/** 接受者 */
@property (nonatomic, strong) Receiver *receiver;
/** 数值 */
@property (nonatomic, assign) CGFloat paramter;
@end

@implementation DarkerCommand


- (instancetype)initWithReceiver:(Receiver*)receiver withParamter:(CGFloat)paramter
{
    self = [super init];
    if (self) {
        self.receiver = receiver;
        self.paramter = paramter;
    }
    return self;
}

/**
 *  命令的执行 思考一下,命令怎么执行让任务实现?
 *
 */
- (void)excute{
    [self.receiver makeViewDarker:self.paramter];
}

/**
 *  撤销命令
 */
- (void)rollBackExcute{
    [self.receiver makeViewLighter:self.paramter];
}

@end

(3)命令的管理者

@interface Invoker : NSObject

interfaceSingleton(Invoker);

/**
 *  添加指令操作
 *
 *  @param command 指令
 */
- (void)addExcute:(id<InvokerProtocol>)command;

/**
 *  回退操作
 */
-(void)rollBack;
@end


@interface Invoker ()
/** 存储指令对象 */
@property (nonatomic,strong)NSMutableArray *commandArray;
@end

@implementation Invoker

implementationSingleton(Invoker);

-(NSMutableArray*)commandArray{
    if (_commandArray == nil) {
        NSLog(@"创建了一次NSMutableArray对象");
        _commandArray = [NSMutableArray array];
    }
    return _commandArray;
}

- (void)addExcute:(id<InvokerProtocol>)command{
    [command excute];
    NSLog(@"开始执行了");
    [self.commandArray addObject:command];
    NSLog(@"执行结束了");
}

-(void)rollBack{
    NSLog(@"撤销操作");
    [self.commandArray.lastObject rollBackExcute];
    [self.commandArray removeLastObject];
}
@end

(4)Receiver任务执行者,和非命令模式下一样

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

推荐阅读更多精彩内容