当前位置: 移动技术网 > IT编程>移动开发>IOS > 设计模式中的Memento备忘录模式的在iOS App开发中的运用

设计模式中的Memento备忘录模式的在iOS App开发中的运用

2019年07月24日  | 移动技术网IT编程  | 我要评论

俄罗斯卫星,主席儿子胡海锋简历,有毒有害工种

备忘录模式。顾名思义,备忘录模式的初衷就是为了返回上一个状态而设计的。从名字看起来一目了然,好吧,还是老样子,先给出定义。

备忘录(memento):在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态。

定义看起来搞的很专业,其实就是保存上一个状态,以便日后恢复用。好比是在玩游戏,在打大boss之前担心第一次打不过,先存个盘,万一玩儿完了,还可以恢复状态重新pk。

下面给出类结构图。

2016323164621128.jpg (500×290)

originator(原发器):记录当前时刻的内部状态,负责定义哪些属于需要备份的状态,负责创建memento,负责从memento恢复状态。

memento(备忘录):负责存储originator的内部状态,在需要时提供给originator内部状态。

caretaker(看管人):将memento保存在安全的地方,并负责提取。

一句话概括:originator创建一个包含其状态的memento交给caretaker保管,caretaker不知如何与memento交互,只负责把memento在安全的地方保存好。
从上面这张图来看,关系比较简单吧。那么备忘录模式一般都用在什么场合呢?

memento模式比较适用于功能比较复杂的,但需要维护或记录属性历史的类,或者需要保存的属性只是众多属性中的一小部分时,originator可以根据保存的memento信息还原到前一状态。有时候一些对象的内部信息必须保存在对象以外的地方,但是必须要由对象自己读取,这时,使用备忘录可以把复杂的对象内部信息对其他的对象屏蔽起来。当然了,最大的作用还是在于当角色的状态改变的时候,有可能这个状态无效,这时候就可以使用暂时存储起来的备忘录将状态进行复原。好啦,其实翻来覆去就是为了恢复数据用的,车轱辘话就不多说了,下面给大家简单展示一下实现的代码吧。

objective-c代码实现:

originator:

复制代码 代码如下:

//发起人:记录当前时刻的内部状态,负责定义哪些属于备份范围的状态,负责创建和恢复备忘录数据。
 
#import <foundation/foundation.h>
@class nimomemento;
 
@interface nimooriginator : nsobject
 
@property (nonatomic, copy) nsstring* state;
 
- (nimomemento *)creatememento;
- (void)restorememento:(nimomemento *)memento;
 
@end

复制代码 代码如下:

#import "nimooriginator.h"
#import "nimomemento.h"
 
@implementation nimooriginator
 
- (nimomemento *)creatememento
{
    nimomemento *memento = [[nimomemento alloc] initwithstate:_state];
    return memento;
}
 
- (void)restorememento:(nimomemento *)memento
{
    _state = memento.state;
}
 
- (nsstring *)description
{
    return [nsstring stringwithformat:@"state:%@", _state];
}
 
@end

memento:
复制代码 代码如下:

//备忘录:负责存储发起人对象的内部状态,在需要的时候提供发起人需要的内部状态。
 
#import <foundation/foundation.h>
 
@interface nimomemento : nsobject
 
@property (nonatomic, copy, readonly) nsstring *state;
- (id)initwithstate:(nsstring *)state;
 
@end

复制代码 代码如下:

#import "nimomemento.h"
 
@interface nimomemento()
 
@property (nonatomic, copy, readwrite) nsstring *state;
 
 
@end

复制代码 代码如下:

@implementation nimomemento
 
- (id)initwithstate:(nsstring *)state
{
    if (self = [super init]) {
        _state = [state copy];
    }
    
    return self;
}
 
@end


caretaker:
复制代码 代码如下:

//管理角色:对备忘录进行管理,保存和提供备忘录。
 
#import <foundation/foundation.h>
@class nimomemento;
 
@interface nimocaretaker : nsobject
 
@property (nonatomic, assign) nimomemento *memento;
 
@end

复制代码 代码如下:

//
//  nimocaretaker.m
//  mementodemo
//

#import "nimocaretaker.h"
 
@implementation nimocaretaker
 
@end


复制代码 代码如下:

client:
 
#import <foundation/foundation.h>
#import "nimooriginator.h"
#import "nimomemento.h"
#import "nimocaretaker.h"
 
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        
        nimooriginator *originator = [[nimooriginator alloc] init];
        originator.state = @"old";
        nslog(@"%@", originator);
        nimomemento *memento = originator.creatememento;
        
        nimocaretaker *caretaker = [[nimocaretaker alloc] init];
        caretaker.memento = memento;
        originator.state = @"new";
        nslog(@"%@", originator);
        
        [originator restorememento:[caretaker memento]];
        nslog(@"%@", originator);
    }
    return 0;
}

运行:

2015-08-12 20:27:39.184 mementodemo[1160:34914] state:old
2015-08-12 20:27:39.186 mementodemo[1160:34914] state:new
2015-08-12 20:27:39.186 mementodemo[1160:34914] state:old

以上通用代码运行后虽然能得到期望的结果,但是并不完美,在menmento类的实现中,我们把state属性以及initwithstate初始化方法暴露在了公共接口中,这两者本应只提供给originator与menmento(即对originator与menmento提供宽接口,对caretaker等其他对象提供窄接口)。在c++等其他面向对象语言中,一般使用private或friend进行声明。但在objective-c中一切都是公有的,所以需要额外的技巧来实现。

通过类扩展将state属性以及initwithstate初始化方法从主接口头文件nimomemento.h中分离:

复制代码 代码如下:

//
//  nimomemento+private.h
//  mementodemo
//
 
#import "nimomemento.h"
 
@interface nimomemento ()
 
@property (nonatomic, copy, readwrite) nsstring *state;
- (id)initwithstate:(nsstring *)state;
 
@end

如此,只在originator与menmento中#import nimomemento+private.h,便实现了接口的私有化。

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

  • ios uicollectionview实现横向滚动

    现在使用卡片效果的app很多,之前公司让实现一种卡片效果,就写了一篇关于实现卡片的文章。文章最后附有demo实现上我选择了使用uicollectionview ... [阅读全文]
  • iOS UICollectionView实现横向滑动

    本文实例为大家分享了ios uicollectionview实现横向滑动的具体代码,供大家参考,具体内容如下uicollectionview的横向滚动,目前我使... [阅读全文]
  • iOS13适配深色模式(Dark Mode)的实现

    iOS13适配深色模式(Dark Mode)的实现

    好像大概也许是一年前, mac os系统发布了深色模式外观, 看着挺刺激, 时至今日用着也还挺爽的终于, 随着iphone11等新手机的发售, ios 13系统... [阅读全文]
  • ios 使用xcode11 新建项目工程的步骤详解

    ios 使用xcode11 新建项目工程的步骤详解

    xcode11新建项目工程,新增了scenedelegate这个类,转而将原appdelegate负责的对ui生命周期的处理担子接了过来。故此可以理解为:ios... [阅读全文]
  • iOS实现转盘效果

    本文实例为大家分享了ios实现转盘效果的具体代码,供大家参考,具体内容如下demo下载地址: ios转盘效果功能:实现了常用的ios转盘效果,轮盘抽奖效果的实现... [阅读全文]
  • iOS开发实现转盘功能

    本文实例为大家分享了ios实现转盘功能的具体代码,供大家参考,具体内容如下今天给同学们讲解一下一个转盘选号的功能,直接上代码直接看viewcontroller#... [阅读全文]
  • iOS实现轮盘动态效果

    本文实例为大家分享了ios实现轮盘动态效果的具体代码,供大家参考,具体内容如下一个常用的绘图,主要用来打分之类的动画,效果如下。主要是ios的绘图和动画,本来想... [阅读全文]
  • iOS实现九宫格连线手势解锁

    本文实例为大家分享了ios实现九宫格连线手势解锁的具体代码,供大家参考,具体内容如下demo下载地址:效果图:核心代码://// clockview.m// 手... [阅读全文]
  • iOS实现卡片堆叠效果

    本文实例为大家分享了ios实现卡片堆叠效果的具体代码,供大家参考,具体内容如下如图,这就是最终效果。去年安卓5.0发布的时候,当我看到安卓全新的material... [阅读全文]
  • iOS利用余弦函数实现卡片浏览工具

    iOS利用余弦函数实现卡片浏览工具

    本文实例为大家分享了ios利用余弦函数实现卡片浏览工具的具体代码,供大家参考,具体内容如下一、实现效果通过拖拽屏幕实现卡片移动,左右两侧的卡片随着拖动变小,中间... [阅读全文]
验证码:
移动技术网