緩存
- plist(主要是系統(tǒng)設(shè)置)
- Cache緩存 (保存在磁盤(pán)、內(nèi)存中)
NSCache
1.NSCache 簡(jiǎn)單介紹
1)NSCache是蘋(píng)果官方提供的緩存類,具體使用和NSMutableDictionary類似,在AFN和SDWebImage框架中被是用來(lái)管理緩存
2)NSCache在系統(tǒng)內(nèi)存很低時(shí),會(huì)自動(dòng)釋放對(duì)象,接收到內(nèi)存警告時(shí)主動(dòng)調(diào)用removeAllobject方法釋放對(duì)象
3)NSCache是線程安全的,在多線程操作中,不需要對(duì)NSCache加鎖
2.NSCache屬性和方法介紹
1)屬性介紹:
name:名稱
delegat:設(shè)置代理
totalCostLimit:緩存控件的最大總成本,超出上限會(huì)自動(dòng)回收對(duì)象,默認(rèn)值為0,表示沒(méi)有限制
countLimit:能夠緩存的對(duì)象的最大數(shù)量:默認(rèn)值為0,表示沒(méi)有限制
evicrsObjectsWithDiscardedContent:標(biāo)識(shí)緩存是否回收廢棄的內(nèi)容
2)方法介紹:
在緩存中設(shè)置指定鍵名對(duì)應(yīng)的值,0成本
-(void)setObject:(ObjectType)obj forkey:(KeyType)key;
在緩存中設(shè)置指定鍵名對(duì)應(yīng)的值,并且指定該鍵值對(duì)的成本,用于計(jì)算記錄在緩存中的所有對(duì)象的總成本,當(dāng)出現(xiàn)內(nèi)存警告或者超出緩存總成本上限的時(shí)候,緩存會(huì)開(kāi)啟一個(gè)回收過(guò)程,刪除部分元素
-(void) setObject:(ObjectType)obj forKey:(keyType)keycost:(NSUInteger)g;
刪除緩存中指定鍵名的對(duì)象
-(void)removeObjcetForKey:(keyType)key;
刪除緩存中所有的對(duì)象
-(void)removeAllObjects;
YYCache
1.簡(jiǎn)書(shū)漢斯哈哈哈
http://www.itdecent.cn/p/b8dcf6634fab
2.YYCache 基于sqlite3.0
(1)SQLite輕量級(jí)的數(shù)據(jù)庫(kù) 使用原生的C函數(shù)庫(kù) 使用時(shí)添加libsqlite3.dylib 接著在使用SQLite API的Object-C類中使用 導(dǎo)入頭文件 #import<sqlite3.h>
(2)數(shù)據(jù)存取讀出:
** 存:存入內(nèi)存時(shí)同時(shí)寫(xiě)入磁盤(pán)。
** 讀:
1)先從內(nèi)存中讀,有就直接拿來(lái)用;
2)內(nèi)存沒(méi)有再?gòu)拇疟P(pán)讀,磁盤(pán)有就直接拿來(lái)用,并寫(xiě)入內(nèi)存;
3)內(nèi)存和磁盤(pán)都沒(méi)有,返回空。
YYMemoryCache
http://www.itdecent.cn/p/492c3c3a0485
實(shí)現(xiàn)原理
YYMemoryCache是內(nèi)存緩存,存取速度非???。但緩存的容量是有限的,當(dāng)Cache占滿后,再緩存數(shù)據(jù)就失敗,必須選擇一個(gè)緩存快來(lái)替換掉已經(jīng)緩存了的塊。
1.LRU 應(yīng)運(yùn)而生
LRU 根據(jù)各塊使用情況,總是選擇那個(gè)最長(zhǎng)時(shí)間未被使用的塊替換。
LRU 縮寫(xiě)為L(zhǎng)east Recently Used 即最近最少使用 它是一種頁(yè)面置換算法同時(shí)也是一種緩存機(jī)制。
2.LRU采用兩種數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)
雙向鏈表
哈希表
3.Cache操作
插入:當(dāng)Cache未滿時(shí),新的數(shù)據(jù)項(xiàng)只需插入到雙鏈表頭部即可
查找:每次數(shù)據(jù)項(xiàng)被查詢到,都將此數(shù)據(jù)項(xiàng)移動(dòng)到鏈表頭部
替換:當(dāng)Cache滿時(shí),將新的數(shù)據(jù)項(xiàng)查到雙鏈表頭部,并刪除雙鏈表的尾結(jié)點(diǎn)。
使用
1.圖標(biāo)--最常用的圖標(biāo)顯示
內(nèi)存緩存的處理思路
1)當(dāng)圖片下載完成后,除了在TableView顯示外,還要保存一份在內(nèi)存緩存中
2)當(dāng)圖片需要展示的時(shí)候,先查看是否已經(jīng)下載過(guò),如果已經(jīng)下載過(guò),有緩存,如果有那么直接使用緩存里的數(shù)據(jù),否則下載下來(lái)用
3)可以用字典和數(shù)組的方式實(shí)現(xiàn)
示例代碼
- (NSMutableDictionary *)imageCache{
if (_imageCache ==nil) {
_imageCache = [NSMutableDictionary dictionary];
}
return _imageCache;
}
磁盤(pán)緩存的處理思路
1)當(dāng)圖片下載完成后,除了保存一份在內(nèi)存緩存中,還要保存一份到沙盒中,也就是磁盤(pán)中
2)查看是否有磁盤(pán)緩存,如果有那么使用磁盤(pán)磁盤(pán),保存一份到內(nèi)存中
//1.得到cache路徑
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject];
//2.文件名稱
NSString *fileName = [appM.icon lastPathComponent];
//3.拼接文件的全路徑
NSString *fullpath = [caches stringByAppendingPathComponent:fileName];
//4.嘗試查看是否有磁盤(pán)緩存
NSData *data = [NSData dataWithContentsOfFile:fullpath];
//5.廢棄磁盤(pán)緩存
data = nil;
if (data) {
UIImage *image = [UIImage imageWithData: data];
cell.imageView.image = image;
//6.保存一份到內(nèi)存中
[self.images setObject:image forKey:appM.icon];
NSLog(@"%zd行 對(duì)應(yīng)的圖片 從 磁盤(pán) 中加載",indexPath.row);
2.當(dāng)發(fā)生內(nèi)存警告的時(shí)候 手動(dòng)清理內(nèi)存緩存
[self.image removeAllObject];
3.當(dāng)下載圖片可能會(huì)出現(xiàn)的問(wèn)題
(1)UI卡頓 所有的下載都在主線程中處理,解決辦法 開(kāi)子線程下載圖片
(2)圖片重復(fù)下載(用戶上下滾動(dòng)) 內(nèi)存緩存--->優(yōu)化(磁盤(pán)緩存)
(3)當(dāng)網(wǎng)速很慢的時(shí)候,圖片的下載操作會(huì)被添加到隊(duì)列中多次,當(dāng)內(nèi)存緩存和磁盤(pán)緩存中圖片都不存在的時(shí)候,先檢查圖片的下載操作是否存在,如果存在那么等待即可,如果圖片的下載操作不存在,那么再去封裝操作去下載該圖片-->操作緩存
(4)數(shù)據(jù)顯示錯(cuò)亂
先清空cell的圖片或者設(shè)置占位符
(5)url不正確等數(shù)據(jù)問(wèn)題
下載完圖片之后先判斷圖片是否有值,有就直接返回或者從操作緩存中刪除