(-) 提出問(wèn)題
我們?cè)诰幊痰臅r(shí)候,經(jīng)常需要保存對(duì)象的一個(gè)狀態(tài),當(dāng)需要的時(shí)候,可以恢復(fù)到這個(gè)狀態(tài)。比如,我們使用Xcode進(jìn)行編程時(shí),假如編寫(xiě)失誤(例如不小心誤刪除了幾行代碼),我們希望返回刪除前的狀態(tài),便可以使用Ctrl+Z來(lái)進(jìn)行返回。這時(shí)我們便可以使用備忘錄模式來(lái)實(shí)現(xiàn)。
(二 )備忘錄設(shè)計(jì)模式 簡(jiǎn)介
在不破壞封裝性的前提下,捕獲一個(gè)對(duì)象的內(nèi)部狀態(tài),并在該對(duì)象之外保存這個(gè)狀態(tài),這樣以后就可以將對(duì)象恢復(fù)到原先保存的的狀態(tài)?!对O(shè)計(jì)模式》
通用類(lèi)圖如下:

使用場(chǎng)景:
當(dāng)滿(mǎn)足以下兩個(gè)條件的時(shí)候,應(yīng)當(dāng)考慮使用這一模式。
1、需要保存一個(gè)對(duì)象(活某部分)在某一時(shí)刻的狀態(tài),這樣以后就可以恢復(fù)先前的狀態(tài);
2、用于獲取狀態(tài)的接口,會(huì)暴露實(shí)現(xiàn)的細(xì)節(jié),需要將其隱藏起來(lái);
(二 )使用備忘錄設(shè)計(jì)模式
通過(guò)一個(gè)簡(jiǎn)單的例子來(lái)實(shí)現(xiàn)這個(gè)設(shè)計(jì)模式。例子很簡(jiǎn)單,只是完成一個(gè)類(lèi)的一個(gè)屬性保存和恢復(fù)。
第1步 創(chuàng)建協(xié)議
創(chuàng)建協(xié)議的原因是因?yàn)閭渫洿鎯?chǔ)的數(shù)據(jù)不一定是什么類(lèi)型的,我們要讓備忘錄能存儲(chǔ)很多類(lèi)型的數(shù)據(jù),需要讓需要備忘的類(lèi)實(shí)現(xiàn)一套統(tǒng)一的接口。
協(xié)議包含兩個(gè)方法,一個(gè)獲取當(dāng)前狀態(tài)的方法,和一個(gè)恢復(fù)狀態(tài)的方法。需要使用備忘錄的類(lè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)建備忘錄類(lèi)
備忘錄中心類(lèi)只負(fù)責(zé)數(shù)據(jù)的存儲(chǔ)和恢復(fù)。傳過(guò)來(lái)的object必須是繼承上面協(xié)議的對(duì)象,這樣就可以調(diào)用協(xié)議里的方法獲得需要備忘的數(shù)據(jù)了。類(lèi)里有兩個(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è)類(lèi)FastCoder的方法,F(xiàn)astCoder是一個(gè)Cocoa object和object graph的高性能二進(jìn)制序列化格式,可以作為Property Lists和JSON的替代選擇。FastCoder git地址。
第3步 創(chuàng)建工具類(lèi)
如果我們直接調(diào)用備忘錄的方法的話(huà),會(huì)暴露出我們實(shí)現(xiàn)的細(xì)節(jié),也會(huì)暴露出我們要備忘的object。所以我們創(chuàng)建一個(gè)工具類(lèi),來(lái)再封裝一層。這樣我們就不會(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è)試類(lèi)
測(cè)試類(lèi)里只有一個(gè)字符串屬性,我們來(lái)恢復(fù)這個(gè)屬性。測(cè)試類(lèi)要實(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é)。
(四)鳴謝
這篇文章是看過(guò) YouXianMing老師 的教程后,得到的啟發(fā)并重構(gòu)自己項(xiàng)目代碼后寫(xiě)的總結(jié)。希望對(duì)大家有點(diǎn)幫助。在此感謝YouXianMing老師。