(-) 提出問題
我們?cè)诰幊痰臅r(shí)候曼验,經(jīng)常需要保存對(duì)象的一個(gè)狀態(tài)勾拉,當(dāng)需要的時(shí)候,可以恢復(fù)到這個(gè)狀態(tài)形入。比如全跨,我們使用Xcode進(jìn)行編程時(shí),假如編寫失誤(例如不小心誤刪除了幾行代碼)亿遂,我們希望返回刪除前的狀態(tài)浓若,便可以使用Ctrl+Z來進(jìn)行返回。這時(shí)我們便可以使用備忘錄模式來實(shí)現(xiàn)蛇数。
(二 )備忘錄設(shè)計(jì)模式 簡(jiǎn)介
在不破壞封裝性的前提下挪钓,捕獲一個(gè)對(duì)象的內(nèi)部狀態(tài),并在該對(duì)象之外保存這個(gè)狀態(tài)耳舅,這樣以后就可以將對(duì)象恢復(fù)到原先保存的的狀態(tài)碌上。——《設(shè)計(jì)模式》
通用類圖如下:
使用場(chǎng)景:
當(dāng)滿足以下兩個(gè)條件的時(shí)候浦徊,應(yīng)當(dāng)考慮使用這一模式馏予。
1、需要保存一個(gè)對(duì)象(活某部分)在某一時(shí)刻的狀態(tài)盔性,這樣以后就可以恢復(fù)先前的狀態(tài)霞丧;
2、用于獲取狀態(tài)的接口冕香,會(huì)暴露實(shí)現(xiàn)的細(xì)節(jié)蛹尝,需要將其隱藏起來;
(二 )使用備忘錄設(shè)計(jì)模式
通過一個(gè)簡(jiǎn)單的例子來實(shí)現(xiàn)這個(gè)設(shè)計(jì)模式暂筝。例子很簡(jiǎn)單,只是完成一個(gè)類的一個(gè)屬性保存和恢復(fù)硬贯。
第1步 創(chuàng)建協(xié)議
創(chuàng)建協(xié)議的原因是因?yàn)閭渫洿鎯?chǔ)的數(shù)據(jù)不一定是什么類型的焕襟,我們要讓備忘錄能存儲(chǔ)很多類型的數(shù)據(jù),需要讓需要備忘的類實(shí)現(xiàn)一套統(tǒng)一的接口饭豹。
協(xié)議包含兩個(gè)方法鸵赖,一個(gè)獲取當(dāng)前狀態(tài)的方法,和一個(gè)恢復(fù)狀態(tài)的方法拄衰。需要使用備忘錄的類在這兩個(gè)方法里實(shí)現(xiàn)存儲(chǔ)和恢復(fù)的邏輯它褪。
#import <Foundation/Foundation.h>
@protocol MementoCenterProtocol <NSObject>
/*
獲得狀態(tài)
*/
-(id)getStatus;
/*
設(shè)置狀態(tài)
*/
-(void)setStatus:(id)data;
@end
第2步 創(chuàng)建備忘錄類
備忘錄中心類只負(fù)責(zé)數(shù)據(jù)的存儲(chǔ)和恢復(fù)。傳過來的object必須是繼承上面協(xié)議的對(duì)象翘悉,這樣就可以調(diào)用協(xié)議里的方法獲得需要備忘的數(shù)據(jù)了茫打。類里有兩個(gè)方法。
根據(jù)key保存狀態(tài)
/**
* 保存?zhèn)渫?*
* @param object object
* @param key key
*/
+(void)saveMementoObject:(id<MementoCenterProtocol>)object withKey:(NSString*)key{
if (object==nil || key==nil) {
return;
}
id data = [object getStatus];
NSData *tmpData = [FastCoder dataWithRootObject:data];
// 進(jìn)行存儲(chǔ)
if (tmpData) {
[[NSUserDefaults standardUserDefaults] setObject:tmpData
forKey:key];
}
}
根據(jù)key獲得object狀態(tài)數(shù)據(jù)
/**
* 獲得保存的狀態(tài)數(shù)據(jù)
*
* @param key key
*
* @return 狀態(tài)
*/
+(id)getMementObjectWithKey:(NSString*)key{
if (key==nil) {
return nil;
}
id data = nil;
NSData *tmpData = [[NSUserDefaults standardUserDefaults] objectForKey:key];
if (tmpData) {
data = [FastCoder objectWithData:tmpData];
}
return data;
}
這兩個(gè)方法里面用到了一個(gè)類FastCoder的方法,F(xiàn)astCoder是一個(gè)Cocoa object和object graph的高性能二進(jìn)制序列化格式老赤,可以作為Property Lists和JSON的替代選擇轮洋。FastCoder git地址。
第3步 創(chuàng)建工具類
如果我們直接調(diào)用備忘錄的方法的話抬旺,會(huì)暴露出我們實(shí)現(xiàn)的細(xì)節(jié)弊予,也會(huì)暴露出我們要備忘的object。所以我們創(chuàng)建一個(gè)工具類开财,來再封裝一層汉柒。這樣我們就不會(huì)暴露具體的實(shí)現(xiàn)細(xì)節(jié)了。
/**
* 備忘狀態(tài)
*
* @param key 鍵值
*/
- (void)saveStateWithKey:(NSString *)key;
/**
* 恢復(fù)狀態(tài)
*
* @param key 鍵值
*/
- (void)recoverFromStateWithKey:(NSString *)key;
實(shí)現(xiàn):
- (void)saveStateWithKey:(NSString *)key{
if (key==nil) {
return;
}
id<MementoCenterProtocol> obj = (id<MementoCenterProtocol>)self;
if ([obj respondsToSelector:@selector(getStatus)]) {
[MementoCenter saveMementoObject:obj withKey:key];
}
}
- (void)recoverFromStateWithKey:(NSString *)key{
if (key==nil) {
return;
}
id state = [MementoCenter getMementObjectWithKey:key];
id<MementoCenterProtocol> obj = (id<MementoCenterProtocol>)self;
if ([obj respondsToSelector:@selector(setStatus:)]) {
[obj setStatus:state];
}
}
第4步 創(chuàng)建測(cè)試類
測(cè)試類里只有一個(gè)字符串屬性责鳍,我們來恢復(fù)這個(gè)屬性碾褂。測(cè)試類要實(shí)現(xiàn)上面協(xié)議的方法。
-(id)getStatus{
return self.valueStr.length? self.valueStr: @"當(dāng)前運(yùn)動(dòng):10.00KM";
}
-(void)setStatus:(id)data{
if (data) {
self.valueStr = (NSString*)data;
}
}
第5步 測(cè)試
TestView *texeView = [[TestView alloc]initWithFrame:CGRectMake(100, 100, 100, 100)];
[self.view addSubview:texeView];
texeView.valueStr = @"idage";
NSLog(@"恢復(fù)前 ===%@",texeView.valueStr );
//保存
[texeView saveStateWithKey:@"test"];
texeView.valueStr = @"idage163";
//恢復(fù)
[texeView recoverFromStateWithKey:@"test"];
NSLog(@"恢復(fù)后 ===%@",texeView.valueStr );
這樣我們就是實(shí)現(xiàn)了既能保存和恢復(fù)狀態(tài)薇搁,也不會(huì)暴露保存和恢復(fù)的細(xì)節(jié)斋扰。
(四)鳴謝
這篇文章是看過 YouXianMing老師 的教程后,得到的啟發(fā)并重構(gòu)自己項(xiàng)目代碼后寫的總結(jié)啃洋。希望對(duì)大家有點(diǎn)幫助传货。在此感謝YouXianMing老師。