IOS设计模式第九篇之备忘录模式

版权声明:原创作品,谢绝转载!否则将追究法律责任。

备忘录模式捕获和具体化对象的内部状态。换句话说,它可以节省你的东西后来,这种外部状态可以恢复在不违反封装;

也就是说,私人数据是私有的。

怎么用备忘录设计模式

在Viewcontroller的实现文件里添加下面方法:

- (void)saveCurrentState

{

// When the user leaves the app and then comes back again, he wants it to be in the exact same state

// he left it. In order to do this we need to save the currently displayed album.

// Since it's only one piece of information we can use NSUserDefaults.

    [[NSUserDefaults standardUserDefaults] setInteger:currentAlbumIndex forKey:@"currentAlbumIndex"];

}

- (void)loadPreviousState

{

    currentAlbumIndex = [[NSUserDefaults standardUserDefaults] integerForKey:@"currentAlbumIndex"];

    [self showDataForAlbumAtIndex:currentAlbumIndex];

}

saveCurrentState 保存了这个当前专辑的索引用NSUserDefaults 他是一个苹果提供的标准的存储应用设置和数据的。

loadPreviousState 加载之前保存的索引。这不是备忘录设计模式的完全实现。

现在添加代码在滑动视图初始化前面:

[self loadPreviousState];

应用程序启动时加载以前保存的状态,但是你保存应用当前转台只是为了加载?你将要用通知做这件事。当应用进入到后台系统发送

UIApplicationDidEnterBackgroundNotification。你可以用这个通知调用saveCurrentState。这不是很方面吗?

在ViewDidLoad添加下面代码:

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(saveCurrentState) name:UIApplicationDidEnterBackgroundNotification object:nil];

现在这个应用进入到后台。将要自动的调用saveCurrentState方法来保存当前状态。

现在添加下面代码:

- (void)dealloc

{

    [[NSNotificationCenter defaultCenter] removeObserver:self];

}

这个保证你Viewcontroller销毁的时候移除这个类注册的观察者。

构建和运行您的应用程序。导航到一个专辑,将应用程序发送到后台使用命令+ Shift + H(如果你是在模拟器),然后关闭应用程序,重新启动,并检查之前,选择专辑为中心:

看起来你的专辑数据是正确的。但是你的滑动视图没有在中间这是怎么了?

这就是上面可选协议方法的意义

initialViewIndexForHorizontalScroller:因为这个方法在代理类里面没有实现。在这个例子代理类就是Viewcontroller,因此初始视图一直被设置为第一个视图。

为了修正他在Viewcontroller的实现文件里面添加下面代码:

- (NSInteger)initialViewIndexForHorizontalScroller:(HorizontalScroller *)scroller

{

    return currentAlbumIndex;

}

现在这个滑动视图的第一个视图可以设置为任何专辑是由

currentAlbumIndex表明的。这是一个很好的方法来确保你的应用体验来维持私有和可恢复的。

如果你看看PersistencyManager的init,你将要注意到这个专辑数据是硬件编码并且PersistencyManager创建的时候都会重新创建。最好的解决方法是一旦创建专辑列表就把他们存储到一个文件。那么怎么把专辑存储到文件呢?

一种方法是迭代专辑的属性,保存他们到一个plist文件并且然后创建专辑的实例一旦需要的时候。但是这不是最佳选择,由于需要你编写特定的代码依靠每个类的属性和数据。例如你接下来创建的movies类有不同的属性,保存和加载数据需要新的代码。

此外,你不能保存每个类的实例的私有变量,因为他们无法访问外部类。这就是为什么苹果公司创建归档机制。

归档

苹果的一个专门记忆模式的实现是归档。把一个对象转化为流便于保存和恢复不用向外部类暴露私有的属性。你可以参考Apple’s Archives and Serializations Programming Guide.

那么怎么用归档呢?

首先你需要声明专辑通过符合NSCoding协议是可以归档的。在专辑头文件写上符合NSCoding协议:

@interface Album : NSObject <NSCoding>

在实现文件里面:

添加两个方法:

- (void)encodeWithCoder:(NSCoder *)aCoder

{

    [aCoder encodeObject:self.year forKey:@"year"];

    [aCoder encodeObject:self.title forKey:@"album"];

    [aCoder encodeObject:self.artist forKey:@"artist"];

    [aCoder encodeObject:self.coverUrl forKey:@"cover_url"];

    [aCoder encodeObject:self.genre forKey:@"genre"];

}

- (id)initWithCoder:(NSCoder *)aDecoder

{

    self = [super init];

    if (self)

    {

        _year = [aDecoder decodeObjectForKey:@"year"];

        _title = [aDecoder decodeObjectForKey:@"album"];

        _artist = [aDecoder decodeObjectForKey:@"artist"];

        _coverUrl = [aDecoder decodeObjectForKey:@"cover_url"];

        _genre = [aDecoder decodeObjectForKey:@"genre"];

    }

    return self;

}

你调用encodeWithCoder:当你归档你类的一个实例时候,

相反地的当你解档一个实例来创建一个专辑的实例。看着简单而且强大。现在专辑类可以被归档了。

添加的代码实际上保存和加载相册的列表。

添加以下签名(或方法原型)PersistencyManager.h:

这将是叫的方法保存相册。

现在,添加PersistencyManager.m方法实现:

- (void)saveAlbums

{

NSString *filename = [NSHomeDirectory() stringByAppendingString:@"/Documents/albums.bin"];

NSData *data = [NSKeyedArchiver archivedDataWithRootObject:albums];

    [data writeToFile:filename atomically:YES];

}

NSKeyedArchiver档案专辑数组名为albums.bin的文件。

当你归档对象包含其他对象,

归档器会自动的迭代归档子对象并且孩子的任何子对象。

在此情况下开始对专辑类归档,这是专辑实例的一个数组,因此数组和专辑都支持NSCoding接口。任何在数组里面的东西都会被归档。

现在在

PersistencyManager.m init方法里添加下面代码:

- (id)init

{

    self = [super init];

    if (self) {

NSData *data = [NSData dataWithContentsOfFile:[NSHomeDirectory() stringByAppendingString:@"/Documents/albums.bin"]];

        albums = [NSKeyedUnarchiver unarchiveObjectWithData:data];

        if (albums == nil)

        {

            albums = [NSMutableArray arrayWithArray:

                    @[[[Album alloc] initWithTitle:@"Best of Bowie" artist:@"David Bowie" coverUrl:@"http://www.coversproject.com/static/thumbs/album/album_david%20bowie_best%20of%20bowie.png" year:@"1992"],

                    [[Album alloc] initWithTitle:@"It's My Life" artist:@"No Doubt" coverUrl:@"http://www.coversproject.com/static/thumbs/album/album_no%20doubt_its%20my%20life%20%20bathwater.png" year:@"2003"],

                    [[Album alloc] initWithTitle:@"Nothing Like The Sun" artist:@"Sting" coverUrl:@"http://www.coversproject.com/static/thumbs/album/album_sting_nothing%20like%20the%20sun.png" year:@"1999"],

                    [[Album alloc] initWithTitle:@"Staring at the Sun" artist:@"U2" coverUrl:@"http://www.coversproject.com/static/thumbs/album/album_u2_staring%20at%20the%20sun.png" year:@"2000"],

                    [[Album alloc] initWithTitle:@"American Pie" artist:@"Madonna" coverUrl:@"http://www.coversproject.com/static/thumbs/album/album_madonna_american%20pie.png" year:@"2000"]]];

            [self saveAlbums];

        }

    }

    return self;

}

在新的代码里,

NSKeyedUnarchiver 从文件里面加载新的专辑数据,如果他存在。如果他不存在,

立即创建相册数据和保存它在下次加载应用程序。

你还想保存这张专辑每次应用程序进入后台数据。

这可能不是必要的,但如果你后来添加的选项更改相册数据

那么你想要确保您已经保存了所有更改。

添加以下LibraryAPI.h方法签名:

- (void)saveAlbums;

应用通过libraryAPI访问所有的服务,这就是应用怎么让PersitencyManager 知道他需要保存专辑的数据。

现在添加这个方法实现到libraryAPI实现文件里:

- (void)saveAlbums

{

    [persistencyManager saveAlbums];

}

这段代码仅仅通过在调用LibraryAPI保存相册到PersistencyMangaer

。和上面的代码使用LibraryAPI触发保存专辑的数据每当Viewcontroller保存其状态。

编译运行你的应用检查是不是所有东西运行正常。

不幸的是,没有简单的方法来检查数据持久性是正确的。幸运的是你可以在finder里检查模拟器的document文件夹看专辑的数据文件是不是被创建,但是为了检查是不是有改变你不得不添加改变专辑数据的功能。

但是代替改变数据,如果你添加一个选项来删除你的专辑当你不想要的时候。此外你应该有一个撤销的选项一旦你误删了专辑。

这是我们下面介绍的一个设计模式命令设计模式。

上一篇:阿里云实时数仓Hologres年度发布,解读数仓新趋势


下一篇:Java架构师之面试题