今天小編跟大家分享一篇iOS系統(tǒng)緩存的詳細(xì)解析,感興趣的朋友跟小編一起來了解一下吧!
一、關(guān)于同一個(gè)URL的多次請求
有時(shí)候,對同一個(gè)URL請求多次,返回的數(shù)據(jù)可能都是一樣的,比如服務(wù)器上的某張圖片,無論下載多少次,返回的數(shù)據(jù)都是一樣的。
上面的情況會造成以下問題
(1)用戶流量的浪費(fèi)
(2)程序響應(yīng)速度不夠快

解決上面的問題,一般考慮對數(shù)據(jù)進(jìn)行緩存。
二、緩存
為了提高程序的響應(yīng)速度,可以考慮使用緩存(內(nèi)存緩存/硬盤緩存)

第一次請求數(shù)據(jù)時(shí),內(nèi)存緩存中沒有數(shù)據(jù),硬盤緩存中沒有數(shù)據(jù)。
緩存數(shù)據(jù)的過程

當(dāng)服務(wù)器返回?cái)?shù)據(jù)時(shí),需要做以下步驟
(1)使用服務(wù)器的數(shù)據(jù)(比如解析、顯示)
(2)將服務(wù)器的數(shù)據(jù)緩存到硬盤(沙盒)
此時(shí)緩存的情況是:內(nèi)存緩存中有數(shù)據(jù),硬盤緩存中有數(shù)據(jù)。
再次請求數(shù)據(jù)分為兩種情況:
(1)如果程序并沒有被關(guān)閉,一直在運(yùn)行
那么此時(shí)內(nèi)存緩存中有數(shù)據(jù),硬盤緩存中有數(shù)據(jù)。如果此時(shí)再次請求數(shù)據(jù),直接使用內(nèi)存緩存中的數(shù)據(jù)即可
(2)如果程序重新啟動(dòng)
那么此時(shí)內(nèi)存緩存已經(jīng)消失,沒有數(shù)據(jù),硬盤緩存依舊存在,還有數(shù)據(jù)。如果此時(shí)再次請求數(shù)據(jù),需要讀取內(nèi)存中緩存的數(shù)據(jù)。
提示:從硬盤緩存中讀取數(shù)據(jù)后,內(nèi)存緩存中又有數(shù)據(jù)了
三、緩存的實(shí)現(xiàn)
1.說明:
由于GET請求一般用來查詢數(shù)據(jù),POST請求一般是發(fā)大量數(shù)據(jù)給服務(wù)器處理(變動(dòng)性比較大)
因此一般只對GET請求進(jìn)行緩存,而不對POST請求進(jìn)行緩存
在iOS中,可以使用NSURLCache類緩存數(shù)據(jù)
iOS 5之前:只支持內(nèi)存緩存。從iOS 5開始:同時(shí)支持內(nèi)存緩存和硬盤緩存
2.NSURLCache
iOS中得緩存技術(shù)用到了NSURLCache類。
緩存原理:一個(gè)NSURLRequest對應(yīng)一個(gè)NSCachedURLResponse
緩存技術(shù):把緩存的數(shù)據(jù)都保存到數(shù)據(jù)庫中。
3.NSURLCache的常見用法
(1)獲得全局緩存對象(沒必要手動(dòng)創(chuàng)建)NSURLCache *cache = [NSURLCache sharedURLCache];
(2)設(shè)置內(nèi)存緩存的最大容量(字節(jié)為單位,默認(rèn)為512KB)- (void)setMemoryCapacity:(NSUInteger)memoryCapacity;
(3)設(shè)置硬盤緩存的最大容量(字節(jié)為單位,默認(rèn)為10M)- (void)setDiskCapacity:(NSUInteger)diskCapacity;
(4)硬盤緩存的位置:沙盒/Library/Caches
(5)取得某個(gè)請求的緩存- (NSCachedURLResponse *)cachedResponseForRequest:(NSURLRequest *)request;
(6)清除某個(gè)請求的緩存- (void)removeCachedResponseForRequest:(NSURLRequest *)request;
(7)清除所有的緩存- (void)removeAllCachedResponses;
4.緩存GET請求
要想對某個(gè)GET請求進(jìn)行數(shù)據(jù)緩存,非常簡單
復(fù)制代碼 代碼如下:
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
// 設(shè)置緩存策略
request.cachePolicy = NSURLRequestReturnCacheDataElseLoad;
只要設(shè)置了緩存策略,系統(tǒng)會自動(dòng)利用NSURLCache進(jìn)行數(shù)據(jù)緩存
5.iOS對NSURLRequest提供了7種緩存策略:(實(shí)際上能用的只有4種)
復(fù)制代碼 代碼如下:
NSURLRequestUseProtocolCachePolicy // 默認(rèn)的緩存策略(取決于協(xié)議)
NSURLRequestReloadIgnoringLocalCacheData // 忽略緩存,重新請求
NSURLRequestReloadIgnoringLocalAndRemoteCacheData // 未實(shí)現(xiàn)
NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData // 忽略緩存,重新請求
NSURLRequestReturnCacheDataElseLoad// 有緩存就用緩存,沒有緩存就重新請求
NSURLRequestReturnCacheDataDontLoad// 有緩存就用緩存,沒有緩存就不發(fā)請求,當(dāng)做請求出錯(cuò)處理(用于離線模式)
NSURLRequestReloadRevalidatingCacheData // 未實(shí)現(xiàn)
6.緩存的注意事項(xiàng)
緩存的設(shè)置需要根據(jù)具體的情況考慮,如果請求某個(gè)URL的返回?cái)?shù)據(jù):
(1)經(jīng)常更新:不能用緩存!比如股票、彩票數(shù)據(jù)
(2)一成不變:果斷用緩存
(3)偶爾更新:可以定期更改緩存策略 或者 清除緩存
提示:如果大量使用緩存,會越積越大,建議定期清除緩存
四、本地緩存開發(fā)相關(guān)
為了節(jié)約流量,同時(shí)也是為了更好的用戶體驗(yàn),目前很多應(yīng)用都使用本地緩存機(jī)制,其中以網(wǎng)易新聞的緩存功能最為出色。我自己的應(yīng)用也想加入本地緩存的功能,于是我從網(wǎng)上查閱了相關(guān)的資料,發(fā)現(xiàn)總體上說有兩種方法。一種是自己寫緩存的處理,一種是采用ASIHTTPRequest中的ASIDownloadCache。
方法一:一般將服務(wù)器第一次返回的數(shù)據(jù)保存在沙盒里面。這樣在手機(jī)斷網(wǎng)的情況下可以從本地讀取數(shù)據(jù)了。
1.保存到沙盒的代碼:
復(fù)制代碼 代碼如下:
+ (void)saveCache:(int)type andID:(int)_id andString:(NSString *)str;
{
NSUserDefaults * setting = [NSUserDefaults standardUserDefaults];
NSString * key = [NSString stringWithFormat:@"detail-%d-%d",type, _id];
[setting setObject:str forKey:key];
[setting synchronize];
}
2.讀取本地沙盒的代碼
讀取之前首先根據(jù)type和Id判斷本地是否有
復(fù)制代碼 代碼如下:
+ (NSString *)getCache:(int)type andID:(int)_id
{
NSUserDefaults * settings = [NSUserDefaults standardUserDefaults];
NSString *key = [NSString stringWithFormat:@"detail-%d-%d",type, _id];
NSString *value = [settings objectForKey:key];
return value;
}
如果沙盒里面有數(shù)據(jù)
復(fù)制代碼 代碼如下:
NSString *value = [Tool getCache:5 andID:self.QiuTime];
if (value) {
NSDictionary *backdict = [value JSONValue];
if ([backdict objectForKey:@"items"]) {
NSArray *array=[NSArray arrayWithArray:[backdict objectForKey:@"items"]];
for (NSDictionary *qiushi in array) {
QiuShi *qs=[[[QiuShi alloc]initWithDictionary:qiushi] autorelease];
[self.list addObject:qs];
}
}
[self.tableView reloadData];
}
[self.tableView tableViewDidFinishedLoadingWithMessage:@"數(shù)據(jù)全部加載完了.."];
self.tableView.reachedTheEnd = YES;
方法二:使用ASIHTTPRequest和ASIDownloadCache實(shí)現(xiàn)本地緩存
1、設(shè)置全局的Cache
在AppDelegate.h中添加一個(gè)全局變量
復(fù)制代碼 代碼如下:
@interface AppDelegate : UIResponder
{
ASIDownloadCache *myCache;
}
@property (strong, nonatomic) UIWindow *window;
@property (nonatomic,retain) ASIDownloadCache *myCache;
在AppDelegate.m中的- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions方法中添加如下代碼
復(fù)制代碼 代碼如下:
//自定義緩存
ASIDownloadCache *cache = [[ASIDownloadCache alloc] init];
self.myCache = cache;
[cache release];
//設(shè)置緩存路徑
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentDirectory = [paths objectAtIndex:0];
[self.myCache setStoragePath:[documentDirectory stringByAppendingPathComponent:@"resource"]];
[self.myCache setDefaultCachePolicy:ASIOnlyLoadIfNotCachedCachePolicy];
在AppDelegate.m中的dealloc方法中添加如下語句
復(fù)制代碼 代碼如下:
[myCache release];
到這里為止,就完成了全局變量的聲明。
2、設(shè)置緩存策略
在實(shí)現(xiàn)ASIHTTPRequest請求的地方設(shè)置request的存儲方式,代碼如下
復(fù)制代碼 代碼如下:
NSString *str = @"http://....../getPictureNews.aspx";
NSURL *url = [NSURL URLWithString:str];
ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
//獲取全局變量
AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate];
//設(shè)置緩存方式
[request setDownloadCache:appDelegate.myCache];
//設(shè)置緩存數(shù)據(jù)存儲策略,這里采取的是如果無更新或無法聯(lián)網(wǎng)就讀取緩存數(shù)據(jù)
[request setCacheStoragePolicy:ASICachePermanentlyCacheStoragePolicy];
request.delegate = self;
[request startAsynchronous];
3、清理緩存數(shù)據(jù)
我在這里采用的是手動(dòng)清理數(shù)據(jù)的方式,在適當(dāng)?shù)牡胤教砑尤缦麓a,我將清理緩存放在了應(yīng)用的設(shè)置模塊:
復(fù)制代碼 代碼如下:
AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate];
[appDelegate.myCache clearCachedResponsesForStoragePolicy:ASICachePermanentlyCacheStoragePolicy];
這里清理的是ASICachePermanentlyCacheStoragePolicy這種存儲策略的緩存數(shù)據(jù),如果更換其他的參數(shù)的話,即可清理對應(yīng)存儲策略的緩存數(shù)據(jù)。
以上就是關(guān)于iOS系統(tǒng)緩存的詳細(xì)解析了,想必都了解了吧,更多相關(guān)內(nèi)容請繼續(xù)關(guān)注武林技術(shù)頻道。
新聞熱點(diǎn)
疑難解答
圖片精選