iOS開發之文件(分段)下載

1HTTP HEAD方法數組

NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url cachePolicy:0 timeoutInterval:kTimeout];
request.HTTPMethod = @"HEAD";
[NSURLConnection sendAsynchronousRequest:request queue:self.myQueue completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
    NSLog(@"%@", response);
    NSLog(@"---------------");
    NSLog(@"%@", data);
}];

運行測試代碼能夠發現,HEAD方法只是返回資源信息,而不會返回數據體緩存

應用場景:網絡

(1) 獲取資源Mimetype多線程

(2) 獲取資源文件大小,用於端點續傳或多線程下載app

2、使用塊代碼獲取網絡資源大小的方法測試

- (void)fileSizeWithURL:(NSURL *)url completion:(void (^)(long long contentLength))completion
{
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url cachePolicy:0 timeoutInterval:kTimeout];
    request.HTTPMethod = @"HEAD";
    NSURLResponse *response = nil;
    [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:NULL];
   
    completion(response.expectedContentLength);
}

3肯定每次下載數據包的僞代碼實現atom

- (void)downloadFileWithURL:(NSURL *)url
{
    [self fileSizeWithURL:url completion:^(long long contentLength) {
        NSLog(@"文件總大小:%lld", contentLength);       
        // 根據大小下載文件
               while (contentLength > kDownloadBytes) {
            NSLog(@"每次下載長度:%lld", (long long)kDownloadBytes);
            contentLength -= kDownloadBytes;
        }
        NSLog(@"最後下載字節數:%lld", contentLength);
    }];
}

4HTTP Range的示例url

n  經過設置Range能夠指定每次從網路下載數據包的大小線程

n  Range示例code

n  bytes=0-499               從0到499的頭500個字節

n  bytes=500-999          從500到999的第二個500字節

n  bytes=500-                 從500字節之後的全部字節

n  bytes=-500                 最後500個字節

n  bytes=500-599,800-899      同時指定幾個範圍

n  Range小結

n  -   用於分隔

p 前面的數字表示起始字節數

p 後面的數組表示截止字節數,沒有表示到末尾

n  用於分組,能夠一次指定多個Range,不過不多用

5、分段Range代碼實現

long long fromBytes = 0;
long long toBytes = 0;
while (contentLength > kDownloadBytes) {
    toBytes = fromBytes + kDownloadBytes - 1;
    NSString *range = [NSString stringWithFormat:@"bytes=%lld-%lld", fromBytes, toBytes];
    NSLog(@"range %@", range);
    fromBytes += kDownloadBytes;
    contentLength -= kDownloadBytes;
}
fromBytes = fromBytes + contentLength - 1;
NSString *range = [NSString stringWithFormat:@"bytes=%lld-%lld", fromBytes, toBytes];
NSLog(@"range %@", range);

6分段下載文件

/**NSURLRequestUseProtocolCachePolicy = 0 // 默認的緩存策略,內存緩存
 NSURLRequestReloadIgnoringLocalCacheData = 1// 忽略本地的內存緩存*/
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url cachePolicy:NSURLRequestReloadIgnoringCacheData timeoutInterval:kTimeout];
NSString *range = [NSString stringWithFormat:@"bytes=%lld-%lld", from, end];
[request setValue:range forHTTPHeaderField:@"Range"];
NSURLResponse *response = nil;
NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:NULL];
   
NSLog(@"%@-%@-%ld", range, response, (unsigned long)data.length);

提示:

若是GET包含Range請求頭,響應會以狀態碼206(PartialContent)返回而不是200(OK)

7、將數據寫入文件

- (void)appendData:(NSData *)data
{
// 打開緩存文件
NSFileHandle *fp =
[NSFileHandle fileHandleForWritingAtPath:self.cachePath];
// 若是文件不存在,直接寫入數據
if (!fp) {
    [data writeToFile:self.cachePath atomically:YES];
} else {
    // 移動到文件末尾
    [fp seekToEndOfFile];
    // 將數據文件追加到文件末尾
    [fp writeData:data];
    // 關閉文件句柄
    [fp closeFile];
}
}

8檢查文件大小

// 判斷文件是否存在
if ([[NSFileManager defaultManager] fileExistsAtPath:self.cachePath]) {
    NSDictionary *dict = [[NSFileManager defaultManager] attributesOfItemAtPath:self.cachePath error:NULL];
    return [dict[NSFileSize] longLongValue];
} else {
    return 0;
}

提示:因爲數據是追加的,爲了不重複從網絡下載文件,在下載以前

  1. 判斷緩存路徑中文件是否已經存在

  2. 若是存在檢查文件大小

  3. 若是文件大小與網絡資源大小一致,則再也不下載

相關文章
相關標籤/搜索