你好,歡迎來到IOS教程網

 Ios教程網 >> IOS編程開發 >> IOS開發綜合 >> iOS系統緩存方面開發的相關基礎

iOS系統緩存方面開發的相關基礎

編輯:IOS開發綜合

一、關於同一個URL的多次請求
 
  有時候,對同一個URL請求多次,返回的數據可能都是一樣的,比如服務器上的某張圖片,無論下載多少次,返回的數據都是一樣的。

上面的情況會造成以下問題
 
(1)用戶流量的浪費
 
(2)程序響應速度不夠快
 
解決上面的問題,一般考慮對數據進行緩存。
 
 
二、緩存

 
  為了提高程序的響應速度,可以考慮使用緩存(內存緩存\硬盤緩存)

第一次請求數據時,內存緩存中沒有數據,硬盤緩存中沒有數據。
 
緩存數據的過程

當服務器返回數據時,需要做以下步驟
 
(1)使用服務器的數據(比如解析、顯示)
 
(2)將服務器的數據緩存到硬盤(沙盒)
 
此時緩存的情況是:內存緩存中有數據,硬盤緩存中有數據。
 
再次請求數據分為兩種情況:
 
(1)如果程序並沒有被關閉,一直在運行
 
  那麼此時內存緩存中有數據,硬盤緩存中有數據。如果此時再次請求數據,直接使用內存緩存中的數據即可
 
(2)如果程序重新啟動
 
  那麼此時內存緩存已經消失,沒有數據,硬盤緩存依舊存在,還有數據。如果此時再次請求數據,需要讀取內存中緩存的數據。
 
提示:從硬盤緩存中讀取數據後,內存緩存中又有數據了
 
 
 
三、緩存的實現
 
1.說明:
 
由於GET請求一般用來查詢數據,POST請求一般是發大量數據給服務器處理(變動性比較大)
 
因此一般只對GET請求進行緩存,而不對POST請求進行緩存
 
  在iOS中,可以使用NSURLCache類緩存數據
 
  iOS 5之前:只支持內存緩存。從iOS 5開始:同時支持內存緩存和硬盤緩存
 
 
 
2.NSURLCache
 
iOS中得緩存技術用到了NSURLCache類。
 
緩存原理:一個NSURLRequest對應一個NSCachedURLResponse
 
緩存技術:把緩存的數據都保存到數據庫中。
 
 
 
3.NSURLCache的常見用法
 
(1)獲得全局緩存對象(沒必要手動創建)NSURLCache *cache = [NSURLCache sharedURLCache];
 
(2)設置內存緩存的最大容量(字節為單位,默認為512KB)- (void)setMemoryCapacity:(NSUInteger)memoryCapacity;
 
(3)設置硬盤緩存的最大容量(字節為單位,默認為10M)- (void)setDiskCapacity:(NSUInteger)diskCapacity;
 
(4)硬盤緩存的位置:沙盒/Library/Caches
 
(5)取得某個請求的緩存- (NSCachedURLResponse *)cachedResponseForRequest:(NSURLRequest *)request;
 
(6)清除某個請求的緩存- (void)removeCachedResponseForRequest:(NSURLRequest *)request;
 
(7)清除所有的緩存- (void)removeAllCachedResponses;
 
 
 
4.緩存GET請求
 
  要想對某個GET請求進行數據緩存,非常簡單
 復制代碼 代碼如下:
  NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
 
  // 設置緩存策略
 
  request.cachePolicy = NSURLRequestReturnCacheDataElseLoad;
 
  只要設置了緩存策略,系統會自動利用NSURLCache進行數據緩存
 
 
 
5.iOS對NSURLRequest提供了7種緩存策略:(實際上能用的只有4種)
 復制代碼 代碼如下: 
NSURLRequestUseProtocolCachePolicy // 默認的緩存策略(取決於協議)
 
NSURLRequestReloadIgnoringLocalCacheData // 忽略緩存,重新請求
 
NSURLRequestReloadIgnoringLocalAndRemoteCacheData // 未實現
 
NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData // 忽略緩存,重新請求
 
NSURLRequestReturnCacheDataElseLoad// 有緩存就用緩存,沒有緩存就重新請求
 
NSURLRequestReturnCacheDataDontLoad// 有緩存就用緩存,沒有緩存就不發請求,當做請求出錯處理(用於離線模式)
 
NSURLRequestReloadRevalidatingCacheData // 未實現

 
 
6.緩存的注意事項
 
緩存的設置需要根據具體的情況考慮,如果請求某個URL的返回數據:
 
  (1)經常更新:不能用緩存!比如股票、彩票數據
 
  (2)一成不變:果斷用緩存
 
  (3)偶爾更新:可以定期更改緩存策略 或者 清除緩存
 
提示:如果大量使用緩存,會越積越大,建議定期清除緩存

四、本地緩存開發相關
為了節約流量,同時也是為了更好的用戶體驗,目前很多應用都使用本地緩存機制,其中以網易新聞的緩存功能最為出色。我自己的應用也想加入本地緩存的功能,於是我從網上查閱了相關的資料,發現總體上說有兩種方法。一種是自己寫緩存的處理,一種是采用ASIHTTPRequest中的ASIDownloadCache。

方法一:一般將服務器第一次返回的數據保存在沙盒裡面。這樣在手機斷網的情況下可以從本地讀取數據了。
1.保存到沙盒的代碼:
 
復制代碼 代碼如下:

+ (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.讀取本地沙盒的代碼
 
讀取之前首先根據type和Id判斷本地是否有
 
復制代碼 代碼如下:

+ (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; 


如果沙盒裡面有數據
 
復制代碼 代碼如下:

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:@"數據全部加載完了.."]; 
        self.tableView.reachedTheEnd  = YES; 

方法二:使用ASIHTTPRequest和ASIDownloadCache實現本地緩存
 
1、設置全局的Cache
    在AppDelegate.h中添加一個全局變量
復制代碼 代碼如下:

@interface AppDelegate : UIResponder  

    ASIDownloadCache *myCache; 

@property (strong, nonatomic) UIWindow *window; 
@property (nonatomic,retain) ASIDownloadCache *myCache; 

   在AppDelegate.m中的- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions方法中添加如下代碼

復制代碼 代碼如下:
//自定義緩存 
ASIDownloadCache *cache = [[ASIDownloadCache alloc] init]; 
self.myCache = cache; 
[cache release]; 
     
//設置緩存路徑 
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES); 
NSString *documentDirectory = [paths objectAtIndex:0]; 
[self.myCache setStoragePath:[documentDirectory stringByAppendingPathComponent:@"resource"]]; 
[self.myCache setDefaultCachePolicy:ASIOnlyLoadIfNotCachedCachePolicy]; 
  
 
    在AppDelegate.m中的dealloc方法中添加如下語句
 復制代碼 代碼如下:

[myCache release]; 

    到這裡為止,就完成了全局變量的聲明。
 
    2、設置緩存策略
    在實現ASIHTTPRequest請求的地方設置request的存儲方式,代碼如下

復制代碼 代碼如下:

NSString *str = @"http://....../getPictureNews.aspx"; 
NSURL *url = [NSURL URLWithString:str]; 
ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url]; 
//獲取全局變量 
AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate]; 
//設置緩存方式 
[request setDownloadCache:appDelegate.myCache]; 
//設置緩存數據存儲策略,這裡采取的是如果無更新或無法聯網就讀取緩存數據 
[request setCacheStoragePolicy:ASICachePermanentlyCacheStoragePolicy]; 
request.delegate = self; 
[request startAsynchronous]; 

    3、清理緩存數據
 
    我在這裡采用的是手動清理數據的方式,在適當的地方添加如下代碼,我將清理緩存放在了應用的設置模塊:

復制代碼 代碼如下:

AppDelegate *appDelegate = [[UIApplication sharedApplication] delegate]; 
[appDelegate.myCache clearCachedResponsesForStoragePolicy:ASICachePermanentlyCacheStoragePolicy];  
 

    這裡清理的是ASICachePermanentlyCacheStoragePolicy這種存儲策略的緩存數據,如果更換其他的參數的話,即可清理對應存儲策略的緩存數據。

  1. 上一頁:
  2. 下一頁:
蘋果刷機越獄教程| IOS教程問題解答| IOS技巧綜合| IOS7技巧| IOS8教程
Copyright © Ios教程網 All Rights Reserved