iOS 網(wǎng)絡(luò)請求url的緩存策略

url緩存策略

問題原因:
  • 多次重復(fù)請同一個url求造成流量浪費
  • 程序相應(yīng)速度不快
解決辦法
  • 為了提高性能,可以考慮使用緩存(內(nèi)存緩存/硬盤緩存)
  • 客戶端檢測有無內(nèi)存緩存/有無硬盤緩存,如果沒有則請求服務(wù)器
當(dāng)服務(wù)器第一次返回數(shù)據(jù)時,需要做以下步驟
  • 使用服務(wù)器的數(shù)據(jù)(比如解析、顯示)
  • 服務(wù)器的數(shù)據(jù)緩存到硬盤(沙盒)

此時緩存的情況是:內(nèi)存緩存中有數(shù)據(jù),硬盤緩存中有數(shù)據(jù)。

再次請求數(shù)據(jù)分為兩種情況:

(1)如果程序并沒有被關(guān)閉,一直在運行

那么此時內(nèi)存緩存中有數(shù)據(jù),硬盤緩存中有數(shù)據(jù)。如果此時再次請求數(shù)據(jù),直接使用內(nèi)存緩存中的數(shù)據(jù)即可

(2)如果程序重新啟動

那么此時內(nèi)存緩存已經(jīng)消失,沒有數(shù)據(jù),硬盤緩存依舊存在,還有數(shù)據(jù)。如果此時再次請求數(shù)據(jù),需要讀取內(nèi)存中緩存的數(shù)據(jù)。

提示:從硬盤緩存中讀取數(shù)據(jù)后,內(nèi)存緩存中又有數(shù)據(jù)了

緩存實現(xiàn)

  • 說明:

由于GET請求一般用來查詢數(shù)據(jù),POST請求一般是發(fā)大量數(shù)據(jù)給服務(wù)器處理(變動性比較大)
因此一般只對GET請求進行緩存,而不對POST請求進行緩存
在iOS中,可以使用NSURLCache類緩存數(shù)據(jù)
iOS 5之前:只支持內(nèi)存緩存。從iOS 5開始:同時支持內(nèi)存緩存和硬盤緩存

2.NSURLCache

iOS中得緩存技術(shù)用到了NSURLCache類。
緩存原理:一個NSURLRequest對應(yīng)一個NSCachedURLResponse
緩存技術(shù):把緩存的數(shù)據(jù)都保存到數(shù)據(jù)庫中。

3.NSURLCache的常見用法

(1)獲得全局緩存對象(沒必要手動創(chuàng)建)NSURLCache *cache = [NSURLCache sharedURLCache];
(2)設(shè)置內(nèi)存緩存的最大容量(字節(jié)為單位,默認為512KB)- (void)setMemoryCapacity:(NSUInteger)memoryCapacity;
(3)設(shè)置硬盤緩存的最大容量(字節(jié)為單位,默認為10M)- (void)setDiskCapacity:(NSUInteger)diskCapacity;
(4)硬盤緩存的位置:沙盒/Library/Caches
(5)取得某個請求的緩存- (NSCachedURLResponse *)cachedResponseForRequest:(NSURLRequest *)request;
(6)清除某個請求的緩存- (void)removeCachedResponseForRequest:(NSURLRequest *)request;
(7)清除所有的緩存- (void)removeAllCachedResponses;

4.緩存GET請求

要想對某個GET請求進行數(shù)據(jù)緩存,非常簡單

   NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];

// 設(shè)置緩存策略

request.cachePolicy = NSURLRequestReturnCacheDataElseLoad;

只要設(shè)置了緩存策略,系統(tǒng)會自動利用NSURLCache進行數(shù)據(jù)緩存

5.iOS對NSURLRequest提供了7種緩存策略:(實際上能用的只有4種)

NSURLRequestUseProtocolCachePolicy // 默認的緩存策略(取決于協(xié)議)
NSURLRequestReloadIgnoringLocalCacheData // 忽略緩存,重新請求NSURLRequestReloadIgnoringLocalAndRemoteCacheData // 未實現(xiàn)NSURLRequestReloadIgnoringCacheData = NSURLRequestReloadIgnoringLocalCacheData // 忽略緩存,重新請求NSURLRequestReturnCacheDataElseLoad// 有緩存就用緩存,沒有緩存就重新請求NSURLRequestReturnCacheDataDontLoad// 有緩存就用緩存,沒有緩存就不發(fā)請求,當(dāng)做請求出錯處理(用于離線模式)
NSURLRequestReloadRevalidatingCacheData // 未實現(xiàn)

6.緩存的注意事項

緩存的設(shè)置需要根據(jù)具體的情況考慮,如果請求某個URL的返回數(shù)據(jù):

(1)經(jīng)常更新:不能用緩存!比如股票、彩票數(shù)據(jù)

(2)一成不變:果斷用緩存

(3)偶爾更新:可以定期更改緩存策略 或者 清除緩存

提示:如果大量使用緩存,會越積越大,建議定期清除緩存

#import
@interface ViewController : UIViewController
@property (strong, nonatomic) NSURLConnection *connection;
@property (strong, nonatomic) NSURLCache *urlCache;
@property (strong, nonatomic) NSURL *url;
@property (strong, nonatomic) NSMutableURLRequest *request;
 -(IBAction)reloadWebView:(UIButton *)sender;
 @end 
 #####import "ViewController.h"
 @interface ViewController ()
 @end
 @implementation ViewController
 

- (void)viewDidLoad
{
[super viewDidLoad];
NSString *paramURLAsString= @"http://blog.sina.com.cn/u/2526279194";
self.urlCache = [NSURLCache sharedURLCache];
[self.urlCache setMemoryCapacity:1*1024*1024];
//創(chuàng)建一個nsurl
self.url = [NSURL URLWithString:paramURLAsString];
//創(chuàng)建一個請求
self.request=[NSMutableURLRequest requestWithURL:self.url
                                               cachePolicy:NSURLRequestUseProtocolCachePolicy
                               timeoutInterval:30.0f];
[self.myWebView loadRequest:self.request];
}

這個例子中,我們請求url為http://blog.sina.com.cn/u/2526279194的網(wǎng)站。如果這個url被緩存了,我們直接從緩存中獲取數(shù)據(jù),否則從http://blog.sina.com.cn/u/2526279194站點上重新獲取數(shù)據(jù)。我們設(shè)置了緩存大小為1M。

- (IBAction)reloadWebView:(UIButton *)sender {

//從請求中獲取緩存輸出
NSCachedURLResponse *response =[self.urlCache cachedResponseForRequest:self.request];
//判斷是否有緩存
if (response != nil){
    NSLog(@"如果有緩存輸出,從緩存中獲取數(shù)據(jù)");
    [self.request setCachePolicy:NSURLRequestReturnCacheDataDontLoad];
}
[self.myWebView loadRequest:self.request];

self.connection = nil;

NSURLConnection *newConnection = [[NSURLConnection alloc] initWithRequest:self.request
                                                                 delegate:self
                                                         startImmediately:YES];
self.connection = newConnection;}
使用下面代碼,我將請求的過程打印出來
- (void)  connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response{
NSLog(@"將接收輸出");
}
- (NSURLRequest *)connection:(NSURLConnection *)connection
         willSendRequest:(NSURLRequest *)request
        redirectResponse:(NSURLResponse *)redirectResponse{
NSLog(@"即將發(fā)送請求");
return(request);}
- (void)connection:(NSURLConnection *)connection
didReceiveData:(NSData *)data{
NSLog(@"接受數(shù)據(jù)");
NSLog(@"數(shù)據(jù)長度為 = %lu", (unsigned long)[data length]);}
- (NSCachedURLResponse *)connection:(NSURLConnection *)connection
              willCacheResponse:(NSCachedURLResponse *)cachedResponse{
NSLog(@"將緩存輸出");
return(cachedResponse);}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection{
NSLog(@"請求完成");}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error{
NSLog(@"請求失敗");}
@end

第一次打印結(jié)果如下

2013-01-31 15:28:29.923 NSURLCacheDemo[27848:907] 即將發(fā)送請求
2013-01-31 15:28:30.043 NSURLCacheDemo[27848:907] 將接收輸出
2013-01-31 15:28:30.045 NSURLCacheDemo[27848:907] 接受數(shù)據(jù)
2013-01-31 15:28:30.047 NSURLCacheDemo[27848:907] 數(shù)據(jù)長度為 = 30047
2013-01-31 15:28:30.095 NSURLCacheDemo[27848:907] 接受數(shù)據(jù)
2013-01-31 15:28:30.098 NSURLCacheDemo[27848:907] 數(shù)據(jù)長度為 = 3575
2013-01-31 15:28:30.102 NSURLCacheDemo[27848:907] 接受數(shù)據(jù)
2013-01-31 15:28:30.104 NSURLCacheDemo[27848:907] 數(shù)據(jù)長度為 = 1482
2013-01-31 15:28:30.105 NSURLCacheDemo[27848:907] 將緩存輸出
2013-01-31 15:28:30.107 NSURLCacheDemo[27848:907] 請求完成
第二次點擊打印結(jié)果如下
2013-01-31 15:28:31.599 NSURLCacheDemo[27848:907] 如果有緩存輸出,從緩存中獲取數(shù)據(jù)
2013-01-31 15:28:31.607 NSURLCacheDemo[27848:907] 即將發(fā)送請求
2013-01-31 15:28:31.840 NSURLCacheDemo[27848:907] 將接收輸出
2013-01-31 15:28:31.843 NSURLCacheDemo[27848:907] 接受數(shù)據(jù)
2013-01-31 15:28:31.845 NSURLCacheDemo[27848:907] 數(shù)據(jù)長度為 = 35104
2013-01-31 15:28:31.846 NSURLCacheDemo[27848:907] 請求完成
我們看到?jīng)]有“將緩存輸出”一項,請求到的數(shù)據(jù)是第一次請求的累積,也就是第二次是從內(nèi)存中獲取數(shù)據(jù)的。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容