美文网首页iOS地图iOS专攻资源__网络专题iOS直播视频
iOS 基础--大文件的下载(断点续传)

iOS 基础--大文件的下载(断点续传)

作者: 云之君兮鹏 | 来源:发表于2016-11-05 21:30 被阅读1835次
古人学问无遗力,少壮工夫老始成!<佛烈托斯>

准备写一个下载的基础总结,发现 简·友 【 xx_cc】 这篇总结写的很好了大家可以一起看看,我分享其中我也用过的方法分析一个下载的小Demo!哎,暂时不干 iOS 抽时间和大家一起学习,有错误地方还请大家指正!GitHub


效果图:


大文件下载.gif
  • 使用NSURLSessionDataTask、NSURLSessionDataDelegate下载大文件思路:

1、下载的时候直接把下载的文件放到沙盒中,可以避免零时变量占用较大的内存同时不会因为中断下载而导致前面下载的内容丢失,这里直接用输出流将数据不断的放到指定沙盒位置!
2、为了实现断点续传,创建下载任务的时候会根据 URL 去找对象的沙盒位置中有没有该文件

  • 如果 有 判断下载多少了,如果是全部下载完成就不需要再去下载,要是下了一部分就设置从已经下载好了之后开始下载。
  • 如果没有的话,直接开始下载并且把文件的总大小记录到沙盒中,以供相应的判断

3、这里存储位置我们可以自己决定,但是文件的名字主要是要和 URL 一一对应,我这里使用下载地址 MD5 转化之后的字符串 加上自己的后缀(类型) 作为文件名。这样每次去比较的的时候就是使用 URL 进行相应的转换即可!

  • 补充一个流的概念

Stream 翻译成为流,它是对我们读写文件的一个抽象,是把文件的内容,一小段一小段的读出或 写入,来到达这样的效果

  • NSStream
    NSStream 是Cocoa平台下对流这个概念的实现类, NSInputStream 和 NSOutputStream 则是它的两个子类,分别对应了读文件和 写文件。
  • NSInputStream
    NSInputStream 对应的是读文件,所以要记住它是要将文件的内容读到内存(你声明的一段buffer)里
  • NSOutputStream
    NSOutputStream 对应的是写文件,它是要将已存在的内存(buffer)里的数据写入文件

代码部分:

1、遵循代理及相关的属性

@interface PP_DownLoad ()<NSURLSessionDataDelegate>

@property(nonatomic,strong)NSOutputStream *stream;// 输出流(对应写入文件)
@property(nonatomic,assign)NSInteger totalLength;// 文件总大小
@property(nonatomic,assign)NSInteger currentLength;// 已经下载大小
@property(nonatomic,strong)NSURLSession *session;
@property(nonatomic,strong)NSURLSessionDataTask *dataTask;
@property (nonatomic, strong) NSString *fileName; // 文件保存名字
@property (nonatomic, strong) NSString *urlString; // 下载路径
@property (nonatomic, strong) NSString *fileLengthName; // 存储文件长度的名字

@end```

2、根据URL创建 任务
```code
-(NSURLSessionDataTask *)dataTaskWithUrlStr:(NSString *)urlString
{
    /*
     * 先去看看已经下载了多少,然后设置从已经下载之后的开始下载!
     */
    if (_dataTask == nil) {
        self.urlString = urlString;
        self.currentLength = [self getCurrent];
        NSURL *url =[NSURL URLWithString:urlString];
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
        NSString *range =[NSString stringWithFormat:@"bytes=%zd-",self.currentLength];// %zd表示 size_t类型进行输出
        [request setValue:range forHTTPHeaderField:@"Range"];
        self.dataTask = [self.session dataTaskWithRequest:request];
        
    }
    return self.dataTask;
}```
3、 计算对应下载的文件大小
```code
-(NSInteger )getCurrent
{
    NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
    NSString *filePath = [caches stringByAppendingPathComponent:self.fileName];
    NSFileManager *manager = [NSFileManager defaultManager];
    NSDictionary *dict = [manager attributesOfItemAtPath:filePath error:nil];
    return [dict[@"NSFileSize"] integerValue];
}```
4、确定存储文件名称  用下载地址 MD5 转化之后的字符串  加上自己的后缀 作为文件名
```code
- (NSString *)fileName
{
    NSArray *prefix_Suffix = [_urlString componentsSeparatedByString:@"."];
    _fileName = [[self getMD5String:_urlString] stringByAppendingFormat:@".%@",[prefix_Suffix lastObject]];
    
    return _fileName;
}
- (NSString *)fileLengthName
{
    return [NSString stringWithFormat:@"%@.txt",self.fileName];
}```

5、 把字符串转化成 MD5字符串 去掉特殊的标记
```code
- (NSString *)getMD5String:(NSString *)string
{
    // 转成 C 语言的字符串
    const char *mdData = [string UTF8String];
    unsigned char result[CC_MD5_DIGEST_LENGTH];
   
    CC_MD5(mdData, (CC_LONG)strlen(mdData), result);
    
    // 化成 OC 可变 字符串
    NSMutableString *mdString  = [NSMutableString new];
    for (int i =0 ; i < CC_MD5_DIGEST_LENGTH; i++)
    {
        [mdString appendFormat:@"%02X",result[i]];
    }
    return mdString;
}```

##### 下载部分:
1、 设置代理
```code
-(NSURLSession *)session
{
    if (_session == nil) {
        // 使用代理方法请求
        /**
         参数一:配置信息
         参数二:代理
         参数三:控制代理方法在那个队列中调用
         遵守代理:NSURLSessionDataDelegate
         */
        _session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
    }
    return _session;
}```

2、设置文件总数
```code
-(void)saveTotal:(NSInteger )length
{
    NSLog(@"开始存储文件大小");
    NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
    NSString *filePath = [caches stringByAppendingPathComponent:self.fileLengthName];
    // 把下载文件的总大小  存在沙盒的缓存里面
    
    NSMutableDictionary *dict = [NSMutableDictionary dictionary];
    [dict setObject:@(length) forKey:self.fileLengthName];
    [dict writeToFile:filePath atomically:YES];
}```

####下载代理部分
- 接收到服务器响应的时候调用
```code
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
{
    // 拿到文件总大小 获得的是当次请求的数据大小,当我们关闭程序以后重新运行,开下载请求的数据是不同的 ,所以要加上之前已经下载过的内容
    NSLog(@"接收到服务器响应");
    self.totalLength = response.expectedContentLength + self.currentLength;
    
    // 把文件总大小保存的沙盒 没有必要每次都存储一次,只有当第一次接收到响应,self.currentLength为零时,存储文件总大小就可以了
    if (self.currentLength == 0) {
        [self saveTotal:self.totalLength];
    }
    NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
    NSString *filePath = [caches stringByAppendingPathComponent:self.fileName];
    NSLog(@"%@",filePath);
    
    // 创建输出流 如果没有文件会创建文件,YES:会往后面进行追加
    NSOutputStream *stream = [[NSOutputStream alloc] initToFileAtPath:filePath append:YES];
    [stream open];
    self.stream = stream;
    //NSLog(@"didReceiveResponse 接受到服务器响应");
    completionHandler(NSURLSessionResponseAllow);
    
    // 调用自己定义下载类的代理方法  供外界获取下载情况(这儿用了代理和 Block 两个方法一样的目的,就是练练手省的生疏了)
    [self.delegate pp_DownLoad:self startFilePath:self.fileName hasLoadLength:self.currentLength totalLength:self.totalLength];
    self.startBlock(self.fileName,self.currentLength,self.totalLength);
}```
- 接收到服务器返回数据时调用,会调用多次
```code
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
    self.currentLength += data.length;
    // 输出流 写数据
    [self.stream write:data.bytes maxLength:data.length];
    //NSLog(@"下载了百分比---->%f %%",1.0 * self.currentLength / self.totalLength * 100);
    NSLog(@"didReceiveData 接受到服务器返回数据");
    // 回调代理方法
    [self.delegate pp_DownLoad:self progressCurrent:self.currentLength totalLength:self.totalLength];
    self.progressBlock(self.currentLength,self.totalLength);
}```
- 当请求完成之后调用,如果请求失败error有值
```code
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
    // 关闭stream
    [self.stream close];
    self.stream = nil;
    NSLog(@"didCompleteWithError 请求完成");
    // 下载完成回调
    [self.delegate pp_DownLoad:self didCompleteWithError:error];
    self.competeBlock(error);
}```

##### 下载执行 :

1、 开始下载
```code
- (void)startDownLoad:(NSString *)downLoadUrl
{
    [[self dataTaskWithUrlStr:downLoadUrl] resume];
}```
1.1 开始下载包含 Block 回调的方法
```code
- (void)startDownLoad:(NSString *)downLoadUrl
       WithStartBlock:(StartLoadBlock)startBlock
        progressBlock:(ProgressBlock)progressBlock
     didCompleteBlock:(CompleteBlock)competeBlock
{
    [[self dataTaskWithUrlStr:downLoadUrl] resume];
    self.startBlock = startBlock;
    self.progressBlock = progressBlock;
    self.competeBlock = competeBlock;
}```
2、暂停下载
```code
- (void)stopDownLoad
{
    [self.dataTask suspend];
}```

-----------
补充在.h中的自定义的协议和 Block
```code
@protocol PP_DownLoadDelegate <NSObject>

// 开始下载
- (void)pp_DownLoad:(PP_DownLoad *)pp_DownLoad
      startFilePath:(NSString *)filePath
      hasLoadLength:(NSInteger)hasLoadLength
        totalLength:(NSInteger)totalLength;

// 获取下载进度
- (void)pp_DownLoad:(PP_DownLoad *)pp_DownLoad
    progressCurrent:(NSInteger)currentLength
        totalLength:(NSInteger)totalLength;
// 下载完成
- (void)pp_DownLoad:(PP_DownLoad *)pp_DownLoad didCompleteWithError:(NSError *)error;
@end

@property (nonatomic, copy) StartLoadBlock startBlock ; // 开始下载回调
@property (nonatomic, copy) ProgressBlock progressBlock ; // 更新数据回调
@property (nonatomic, copy) CompleteBlock competeBlock ;// 下载完成回调```

---------
先写到这儿,慢慢补充!

相关文章

  • iOS 基础--大文件的下载(断点续传)

    准备写一个下载的基础总结,发现 简·友 【 xx_cc】 这篇总结写的很好了大家可以一起看看,我分享其中我也用过...

  • IOS 断点续传原理浅析(第一篇)

    断点续传概述: 断点续传就是从文件上次中断的地方开始重新下载或上传数据,当下载大文件的时候,如果没有实现断点续传功...

  • iOS-16 断点续传 下载

    断点续传概述: 断点续传就是从文件上次中断的地方开始重新下载或上传数据,当下载大文件的时候,如果没有实现断点续传功...

  • iOS中断点续传(随心所欲下载缓存文件到本地)

    iOS中的断点续传,对于做视频类或下载的来说,比较重要,因为对于大文件,我们不可能迅速下载完成,这时候就需要我们本...

  • 基于Okhttp实现断点下载(续传)和分片下载

    断点下载/续传 断点下载是针对下载大文件需求的一种优化机制,可以从上次下载的断点处继续下载。断点续传原理也相同,只...

  • iOS断点续传

    基于iOS10、realm封装的下载器(支持存储读取、断点续传、后台下载、杀死APP重启后的断点续传等功能)。下载...

  • swift3 iOS断点续传下载工具

    XCDownloadTool for iOS swift3 iOS swift 断点续传下载工具,重启APP恢复临...

  • iOS Session 断点续传出错

    iOS Session 下载 ,在iOS 11 下断点续传总是时好时坏,对比iOS10和iOS11 的 resum...

  • 断点续传

    1.断点续传 定义:我们有时在边聊天边下载大文件时,感觉很卡,这时可以暂停下载任务,聊完天再继续下载文件。这就是断...

  • 更好的Android多线程下载框架2.0

    概述 本篇我们我们就来聊聊多线程下载框架,先聊聊我们框架的特点: 多线程 多任务 断点续传 支持大文件 可以自定义...

网友评论

  • 溜溜leesin:大神你的这个方法可以实现多个文件的下载以及和后台的比对更新吗
    云之君兮鹏: @溜溜leesin 不是大神啦,我只是菜鸟,现在还没有做这了,这个没有对比更新
  • 且行且珍惜_iOS:大神,别拦我,我要转摘了啊,,, :smile:
  • cry_0416:其实在请求报文上加个字段就可以了,bytes=currentLength-,currentlength是已经下载内容的大小,即使关了程序,再次发起请求带上字段 就可以指定的位置请求数据
    Metoo丶淡然:在哪个位置加bytes=currentLength-呀
  • 343cea0f7107:可以的

本文标题:iOS 基础--大文件的下载(断点续传)

本文链接:https://www.haomeiwen.com/subject/nqdnyttx.html