iOS开发:多任务断点下载

1.前言

iOS开发中断点下载功能很常见,网上也有很多框架,本文选择了原生的NSURLSessionNSFileHandle,实现了多任务、大文件的断点下载,保证了较低的内存占用。

2.预览


image

3.设计思路


  • 断点下载方案:

NSMutableData 如果文件很大会出现内存警告
NSURLConnection iOS9之后弃用,没有暂停的方法
NSURLSession 推荐
将下载记录保存到plist,重启应用时就能拿到下载记录,从而继续下载。

  • 文件存储方案:

NSMutableData 如果文件很大会出现内存警告
NSOutputStream 推荐
NSFileHandle 推荐
实际使用发现NSOutputSteamNSFileHandle差别不大,但要注意不再写入时需要调用close方法。
内存占用:

屏幕快照 2018-06-01 上午11.36.50.png

  • 多任务

创建下载任务时我们采用了NSURLSessionDataTask,它是NSURLSessionTask的子类,其拥有只读属性taskIdentifier,若要将其作为任务的唯一标识符需要利用KVC。这里我们没有这么做,而是创建了NSURLSession的分类,为其添加了taskKey属性作为任务的为一标识符。创建任务时将传入的URL的MD5值作为key,并将其作查询下载任务、本地文件缓存、下载记录的为一索引。

4.代码


  • 文件结构

创建下载管理者类MYDownloadManager,考虑到会在多个地方调用下载功能,所以将其设计为单例模式。管理者拥有多个下载任务,每个任务有其唯一的key,所以创建一个保存着多个任务的字典。

@interface MYDownloadManager ()<NSURLSessionDataDelegate>
@property (nonatomic, strong) NSMutableDictionary *downloadDict;    // {Key : md5, Value : <MYDonwload *>}
@end

创建下载类

@interface MYDownload : NSObject
@property (nonatomic, copy) NSString *url;                  // 下载地址
@property (nonatomic, assign) long long downloadedLength;   // 已下载大小
@property (nonatomic, assign) long long totalLength;        // 总大小
@property (nonatomic, strong) NSURLSessionDataTask *task;   // 任务
@property (nonatomic, strong) NSFileHandle *fileHandle;     // 文件句柄
@property (nonatomic, copy) ProgressBlock progressBlock;    // 下载进度回调
@property (nonatomic, copy) StateBlock stateBlock;          // 下载状态回调
@end

下载记录Download.plist结构

屏幕快照 2018-06-01 下午4.31.13.png

  • 开始、恢复下载

我们要从上次结束的位置开始下载,所以需要设置请求头,下载指定范围的文件,设置规则如下:

表示头500个字节:Range: bytes=0-499
表示第二个500字节:Range: bytes=500-999
表示最后500个字节:Range: bytes=-500
表示500字节以后的范围:Range: bytes=500-
同时指定几个范围:Range: bytes=100-199,400-500

- (void)downloadWithUrl:(NSString *)url resume:(BOOL)resume progress:(ProgressBlock)progressBlock state:(StateBlock)stateBlock {
    if (!url.length) {
        return;
    }

    // 将url的md5值作为key
    NSString *key = [url md5];
    long long totalLength = [self getTotalLengthWithKey:key];
    long long downloadedLength = [self getDownloadedLengthWithKey:key];
    
    // 任务已完成
    if (totalLength == downloadedLength && totalLength > 0) {
        if (progressBlock) {
            progressBlock(1.0, downloadedLength, totalLength);
        }
        if (stateBlock) {
            stateBlock(MYDownloadStateComplete);
        }
    }

    // 查询任务是否存在
    MYDownload *download = [self.downloadDict valueForKey:key];
    if (download) {
        
        // 取出下载任务
        if (resume) {
            [download.task resume];
        } else {
            [download.task suspend];
            if (download.stateBlock) {
                download.stateBlock(MYDownloadStateSuspend);
            }
        }
    } else {
        
        // 创建下载任务
        NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:nil];
        
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:url]];
        NSString *rangeValue = [NSString stringWithFormat:@"bytes=%lld-", downloadedLength];
        [request setValue:rangeValue forHTTPHeaderField:@"Range"];
        
        NSURLSessionDataTask *task = [session dataTaskWithRequest:request];
        task.taskKey = key;
        task.url = url;
        
        if (resume) {
            [task resume];
        }
        
        // 创建并保存下载对象
        download = [MYDownload new];
        download.url = url;
        download.task = task;
        download.progressBlock = progressBlock;
        download.stateBlock = stateBlock;
        [self.downloadDict setValue:download forKey:key];
    }
}
  • 实现代理方法NSURLSessionDataDelegate

根据dataTask的taskKey可以得到当前下载对象,从服务器返回的response我们可以得到任务的总大小。开辟缓存文件,创建文件句柄准备写入文件,保存下载记录到plist。

// 收到服务器相应
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(nonnull NSURLResponse *)response completionHandler:(nonnull void (^)(NSURLSessionResponseDisposition))completionHandler {
    NSString *key = dataTask.taskKey;
    NSString *url = dataTask.url;
    NSString *filePath = [self getDownloadedPathWithFileName:response.suggestedFilename];
    
    MYDownload *download = [self.downloadDict valueForKey:key];
    
    // 计算总大小并保存到plist
    long long expectedLength = response.expectedContentLength;
    long long downloadedLength = [self getDownloadedLengthWithKey:key];
    long long totalLength = expectedLength + downloadedLength;
    if (totalLength == 0) {
        if (download.progressBlock) {
            download.progressBlock(0.f, downloadedLength, totalLength);
        }
        if (download.stateBlock) {
            download.stateBlock(MYDownloadStateError);
        }
        return;
    }
    NSDictionary *dict = @{@"TotalLength" : @(totalLength),
                           @"Url" : url,
                           @"FileName" : response.suggestedFilename
                           };
    [self setPlistValue:dict forKey:key];
    
    // 创建NSFileHandle
    NSFileManager *fileManager = [NSFileManager defaultManager];
    if (![fileManager fileExistsAtPath:filePath]) {
        [fileManager createFileAtPath:filePath contents:nil attributes:nil];
    }
    NSFileHandle *fileHandle = [NSFileHandle fileHandleForWritingAtPath:filePath];
    
    // 设置下载对象
    download.totalLength = totalLength;
    download.downloadedLength = downloadedLength;
    download.fileHandle = fileHandle;
    
    completionHandler(NSURLSessionResponseAllow);
}

开始接收数据,利用NSFileHandle将文件写入沙盒,不会导致内存占用过高。

// 收到数据(多次调用)
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
    NSString *key = dataTask.taskKey;

    // 写数据
    MYDownload *download = [self.downloadDict valueForKey:key];
    if (download) {
        [download.fileHandle seekToEndOfFile];
        [download.fileHandle writeData:data];
        
        download.downloadedLength += data.length;
        CGFloat progress = (CGFloat) download.downloadedLength / download.totalLength;
        
        if (download.progressBlock) {
            download.progressBlock(progress, download.downloadedLength, download.totalLength);
        }
        if (download.stateBlock) {
            download.stateBlock(MYDownloadStateDownloading);
        }
    }
}

任务完成、中止时关闭NSFileHandle,移除下载对象

// 任务完成、中止
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    NSString *key = task.taskKey;

    // 关闭写数据流
    MYDownload *download = [self.downloadDict valueForKey:key];
    [download.fileHandle closeFile];
    download.fileHandle = nil;

    if (download.stateBlock) {
        if (error) {
            download.stateBlock(MYDownloadStateError);
        } else {
            download.stateBlock(MYDownloadStateComplete);
        }
    }

    [self.downloadDict removeObjectForKey:key];
}

5.其他


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,427评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,551评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,747评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,939评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,955评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,737评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,448评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,352评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,834评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,992评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,133评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,815评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,477评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,022评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,147评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,398评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,077评论 2 355

推荐阅读更多精彩内容