iOS原生文件的下载

小文件的下载

NSData直接从URL中加载数据

系统提供了方法, 让很多东西都可以直接转换成二进制数据的类型(可以从URL加载,file加载等等,单一数据,直接写入)

// 从提供URL中下载数据
 NSData *data = [NSData dataWithContentsOfURL:
                    [NSURL URLWithString: @"http://b.hiphotos.baidu.com/zhidao/pic/item/728da9773912b31b31187e8c8418367adab4e11b.jpg"]];
// 数据转换成图片
    UIImage *image = [UIImage imageWithData:data    ];
    self.imageView.image = image; 
    // 将图片数据,转换成data数据
    NSData *data1 = UIImageJPEGRepresentation(image, 0);
    // 将data数据,进行保存
    [data1 writeToFile:@"/Users/liujiaxin/Desktop/image.jpeg" atomically:YES];
NSURLConnection如果是小文件的话,直接异步请求便能得到相应的数据.

    NSURL *url = [NSURL URLWithString:@"http://www.people.com.cn/mediafile/pic/20150811/19/7260132349722664171.jpg"];
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
  // 发送请求
    [NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
        // 主队列回调 赋值UI
        self.imageView.image = [UIImage imageWithData:data];
    }];
NSURLSession

IOS7以后不建议使用NSURLConnection进行网络数据处理. [session更加灵活,将操作封装到任务中,通过对任务的操作能够精确的控制操作]NSURLSession, 相当于处理放在会话层执行 (有点大一统的想法, 因为二维码相关的也需要会话)
NSURLSessionTask


    /**
     *  步骤
     *  1、创建task(任务)【任务分为:downloadTask, dataTask(uploadTask)】
     *  2、利用 session 创建 task
     *  3、创建 session
     *  4、执行task   - resume 重新执行
     */
    
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login"];
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    request.HTTPMethod = @"POST";
    request.HTTPBody = [@"username=520it&pwd=520it" dataUsingEncoding:NSUTF8StringEncoding];
    
    NSURLSession *session = [NSURLSession sharedSession];

    // 可以不必创建请求直接用url进行获取,但是只能应用于get请求
    
    NSURLSessionTask *task = [session dataTaskWithRequest:request completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
        NSLog(@"%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]);
    }];  
    [task resume];
NSURLSessionDownloadTask
NSURLSession *session = [NSURLSession sharedSession];
    
    /**
     *  创建下载任务 (该任务启动后便会自己进行下载,并存储的)
     *
     *  @param NSURL 下载那个url 的任务
     *
     *  @block 中的数据 location下载好的文件放在磁盘的位置(会自己创建一个临时文件夹的临时目录)
     *           临时目录存放文件,会不定时的删除, 所以最后将文件转移到Cache中
     *    响应头
     */
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/images/minion_02.png"];
    NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
        NSFileManager *mgr = [NSFileManager defaultManager];
//        [mgr createFileAtPath:[response.suggestedFilename cacheDir] contents:nil attributes:nil];
        // 将文件原存储的的地址 ,转换成新的存储地址, (将沙盒cache地址 用url 进行转换)。
        [mgr moveItemAtURL:location toURL:[NSURL fileURLWithPath:[response.suggestedFilename cacheDir]] error:nil];
        NSLog(@"%@",[response.suggestedFilename cacheDir]); 
    }];
    [task resume];

大文件下载

NSURLConnection,在网络一章中已经说明,对于文件的下载上传管理都是交由代理来完成的.
NSURLConnection的代理方法有:NSURLConnectionDataDelegate
1.接收到响应式调用
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSHTTPURLResponse *)response
2.接收到数据时调用
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data[会调用多次, 每次接受文件是由限制的. 所以会进行多次调用,我们需要将每次接受到的数据进行拼接]
3.连接失败时调用
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
4.接收数据完成时调用
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
在大文件下载之前,我们还需要考虑的是,
1>数据放在应用中,建立一个容器专门存放数据,这样调用虽然方便,但是数据存放在应用中,必定会消耗一定的内存,而且会消耗一定得性能.将读取到的数据,存放在沙盒文件中
2>在沙盒中建立文件,存放数据.(一边下载,一边存放)然后将已经保存的数据进行读取就可以了
第一种方案:

// 建立一个容器数据,用来保存下载后的数据(但由于,数据一致存在于内存,所以非常耗性能)
@property(strong,nonatomic) NSMutableData *data;
- (NSMutableData *)data
{
    if (!_data) {
        _data = [NSMutableData data];
    }
    return _data;
}
- (void)viewDidLoad {
    [super viewDidLoad];
    NSString *urlStr = [ @"http://120.25.226.186:32812" stringByAppendingPathComponent: @"resources/videos/minion_01.mp4"];
    NSURL *url =  [NSURL URLWithString:urlStr];
 // 设置请求   
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
// 设置URLConnection,并且设置代理   
 NSURLConnection *urlCon = [NSURLConnection connectionWithRequest:request delegate:self];   
}

// 代理方法。开始接受相应的时候调用
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSHTTPURLResponse *)response
{
    // 由协议头 取出 接收数据的总长度
    self.AllLength = response.expectedContentLength;
}

// 开始接收到数据的时候调用
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{

    // 拼接数据
    [self.data appendData:data];   
  // 取出当前文件下载长度
    self.currentLength += data.length; 
    // 得到控件sliderView的值
    self.slideView.value = 1.0 * self.currentLength / self.AllLength; 
}
// 失败时调用
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
 NSLog(@"%s",__func__);
    
}

// 完成时调用
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
// 完成时 将拼接数据保存
    [self.data writeToFile:[@"minion_01.mp4" cacheDir] atomically:YES];
}

第二种方案:
沙盒建立文件,存放下载数据

- (void)viewDidLoad {
    [super viewDidLoad];
   
    NSString *urlStr = [ @"http://120.25.226.186:32812" stringByAppendingPathComponent: @"resources/videos/minion_01.mp4"];
    NSURL *url =  [NSURL URLWithString:urlStr];
    
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
    NSURLConnection *urlCon = [NSURLConnection connectionWithRequest:request delegate:self];
    
}
// 代理方法。开始接受相应的时候调用
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSHTTPURLResponse *)response
{
    
    // 由协议头 取出 接收数据的总长度
    self.AllLength = response.expectedContentLength;
// 只有利用manager 才能建立文件       
 NSFileManager *mgr = [NSFileManager defaultManager];
    // 有协议头 得到,接收到文件的名称
         NSString *filePath = [response.suggestedFilename cacheDir];
        NSLog(@"%@",filePath);
        self.filePath = filePath;
        [mgr createFileAtPath:filePath contents:NULL attributes:NULL];   
}
// 开始接收到数据的时候调用
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{

    // 拼接数据
//    [self.data appendData:data];
// 建立文件句柄 --- 主要用于文件数据处理   
 NSFileHandle *handle = [NSFileHandle fileHandleForWritingAtPath:self.filePath];
    // 将后一个数据写在前一个数据的后面
    [handle seekToEndOfFile];
    // 写数据
    [handle writeData:data];
    // 取出当前文件下载长度
    self.currentLength += data.length;
    // 得到控件sliderView的值
    self.slideView.value = 1.0 * self.currentLength / self.AllLength;    
}
// 失败时调用
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
 NSLog(@"%s",__func__);   
}
// 完成时调用
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
// 完成时 将拼接数据保存
//    [self.data writeToFile:[@"minion_01.mp4" cacheDir] atomically:YES];
}

第三种情况:当出现网络情况很差,或者出现断网的情况,这样的话,如果下载到99.9%,然后断网,在进行下载的话,是不是很坑呢.....
所以应该我们需要进行断点续传
断点续传的实现.
协议头,我们需要传入这么一个参数Range: 将这么一个参数设置的话, 就想到与告诉服务请,请求的数据的范围从什么地方开始,所以我们如果告诉服务器范围(文件已经下载的大小),就可以了

- (IBAction)btnClick:(UIButton *)sender {
    // 1.切换按钮图片
    sender.selected = !sender.selected;
    // 2.判断是否是继续下载
    if (sender.selected) {
        // 继续下载
        NSLog(@"继续下载");
        NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_02.mp4"];
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
        
        // 只要设置HTTP请求头的Range属性, 就可以实现从指定位置开始下载
        /*
         表示头500个字节:Range: bytes=0-499
         表示第二个500字节:Range: bytes=500-999
         表示最后500个字节:Range: bytes=-500
         表示500字节以后的范围:Range: bytes=500-
         */
        NSString *range = [NSString stringWithFormat:@"bytes %zd-", self.currentLength]; 
        [request setValue:range forHTTPHeaderField:@"Range"];
        
        self.con = [NSURLConnection connectionWithRequest:request delegate:self];
    }else
    {
        // 暂停
        NSLog(@"暂停");
        [self.con cancel];
    }
}

// 接收到服务器的响应
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSHTTPURLResponse *)response
{
    self.totalLength = response.expectedContentLength;
    // 先去文件中获取下载进度
    
    // 并不是每次都需要创建文件
    // 如果已经下载了一部分, 就继续在上一个文件的后面添加
    if (self.currentLength > 0) {
        return;
    }
    
    
    // 创建容器
    // 创建一个空的文件, 用于保存下载的数据
    NSFileManager *manager = [NSFileManager defaultManager];
    NSString *path = [response.suggestedFilename cacheDir];
    NSLog(@"path = %@", path);
    [manager createFileAtPath:path contents:nil attributes:nil];
    self.path = path;
}

// 接收到服务器返回的数据
// 会调用一次或多次
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{

    // 1.创建一个操作文件的句柄
//    NSFileHandle *handle = [NSFileHandle fileHandleForWritingAtPath:self.path];
    
    // 设置数据写入的位置
    // 只要调用这个方法, 每次写入数据都会写入到上一次的后面
    [self.handle seekToEndOfFile];
    
    // 2.利用句柄往文件中写入数据
    [self.handle writeData:data];
    
    // 3.计算当前接收到得数据的总数
    self.currentLength += data.length;
    
    // 计算下载比例
    self.progressView.progress = 1.0 * self.currentLength/self.totalLength;
}

// 接收完毕
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    [self.handle closeFile];
    self.handle = nil;
    
    NSLog(@"下载完毕");
}
#pragma mark - lazy 
- (NSFileHandle *)handle
{
    if (!_handle) {
        // 1.创建一个操作文件的句柄
        _handle = [NSFileHandle fileHandleForWritingAtPath:self.path];
    }
    return _handle;
}

第四种方式 : 句柄 (太陌生), 输出数据流进行数据的存储(根据文件创建数据输出流, 数据流,只接受byte数据 )


// 设置按钮的监听
- (IBAction)clickBtn:(UIButton *)sender;
// 创建网络请求连接
@property (strong, nonatomic) NSURLConnection *con;
// 记录接收文件的总大小
@property (assign, nonatomic) NSInteger totalLength;
// 记录接收到的当前文件的大小
@property (assign, nonatomic) NSInteger currentLength;
// 记录文件存储路径
@property (copy, nonatomic) NSString *toPath;
// 设置输出流,用于接收数据
@property (strong, nonatomic)  NSOutputStream *stream;
@end

@implementation ViewController

- (IBAction)clickBtn:(UIButton *)sender {
    
    sender.selected = !sender.selected;

    if (sender.selected) {
        
        NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_02.mp4"];
        
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
        // 拼接请求头
        NSString *range = [NSString stringWithFormat:@"bytes %zd-", self.currentLength];
        [request setValue:range forHTTPHeaderField:@"Range"];
        
      self.con = [NSURLConnection connectionWithRequest:request delegate:self];
        
    }else{
        // 暂停连接
        [self.con cancel];
    
    }
    
}
// 接收到响应
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
    self.totalLength = response.expectedContentLength;
    // 如果有数据便不在创建新的文件
    if (self.currentLength > 0) {
        return;
    }
    NSFileManager *mgr = [NSFileManager defaultManager];
    // suggestFilename:就是取出响应头中 记录接收数据的名字(不是路径哦)
    self.toPath = [response.suggestedFilename cacheDir];
    NSLog(@"%@",self.toPath);
    [mgr createFileAtPath:self.toPath contents:nil attributes:nil];


}

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
    // 数据流的模式,允许拼接。
    self.stream = [NSOutputStream outputStreamToFileAtPath:self.toPath append:YES];
    // 只有打开数据流,才能进行数据的拼接
    [self.stream open];
    
    // 数据流,接收byte数据 长度为:最大接收数据的长度,如果不这么设置的话,接收到的数据是有问题的。
    [self.stream write:data.bytes maxLength:data.length];
    // 设置数据长度
    self.currentLength += data.length;  
    self.progressView.progress = (float)1.0*self.currentLength / self.totalLength;
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
     NSLog(@"%s",__func__);
// 必须和句柄一样 .关闭
    [self.stream close];
}

第四种:NSURLSession
NSURLSessionTask--->普通下载

  /**
     *  参数1、配置,一般为默认配置
     *  参数2、设置代理,谁成为它的代理 遵守协议[代理方法中接受数据]
     *  参数3、代理回调信息 到那个队列(线程)[一般为主队列]
     *  注意 子协议 继承与父协议, 遵守子协议, 就是遵守父协议
     */
    NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/images/minion_02.png"];
// 创建人物,用于下载数据   
 NSURLSessionTask *task = [session dataTaskWithURL:url];
    [task resume];

代理方法对接收到的数据进行处理,遵守的协议是:NSURLSessionDataDelegate

#pragma mark - NSURLSessionDataDelegate
// 接收到响应
/**
 *  接收到响应时调用
 *
 *  @param session
 *  @param dataTask          触发事件的任务
 *  @param response          响应头
 *  @param completionHandler 回调(告诉我们如何接收(是否)数据)
 *    NSURLSessionResponseCancel = 0,   默认情况 为 取消 接收数据
 *    NSURLSessionResponseAllow = 1,    总是允许接收数据
 *    NSURLSessionResponseBecomeDownload = 2  成为下载时 才接收数据
 */
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
{
    //告诉系统,我们始终接收数据 
completionHandler(NSURLSessionResponseAllow);
  // 创建文件,用来接收数据
    NSFileManager *mgr = [NSFileManager defaultManager];
    
    NSString *toPath = [response.suggestedFilename cacheDir];
    self.toPath = toPath;
    NSLog(@"%@",toPath);
    [mgr createFileAtPath:toPath contents:nil attributes:nil];
  
}
// 输出流懒加载
- (NSOutputStream *)stream
{
    if (!_stream) {
        _stream = [NSOutputStream outputStreamToFileAtPath:self.toPath append:YES];
    }
    return _stream;

}
// 接收到数据
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
//这里我们直接用数据输出流来接受数据. 为了避免多次创建输出流,我们利用懒加载,对输出流进行处理
//    NSOutputStream *stream = [NSOutputStream outputStreamToFileAtPath:self.toPath append:YES];
//    
//    self.stream = stream;
    // 记住要打开数据流
    [self.stream open];
    [self.stream write:data.bytes maxLength:data.length];
}
// 接收完成
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
// 关闭输出流
    [self.stream close];
    self.stream = nil;
    // 接收完成 给空间赋值。
    NSData *data = [NSData dataWithContentsOfFile:self.toPath];
    UIImage *image = [UIImage imageWithData:data];
    self.imageView.image = image;
    
    NSLog(@"%s",__func__);
}

NSURLSessionTask可以实现更为简单的断点下载,苹果为我们提供了两个方法: 1.暂停suspend,2.开始resume
, 但调用suspend时, 任务会被暂停.当调用resume是,任务会从暂停的地方再次下载. 省去了NSURLConnection中请求头记录传入数据的麻烦.

在NSURLSession中提供了专门用于包装下载任务的类NSURLSessionDownloadTask----->普通下载

// 设置session会话,专门用来处理任务的执行
 NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];

    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_16.mp4"];
// 利用会话,创建下载任务 
    NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url];
    // 对于不是block创建的任务,不会立即执行, 要调用resume
    [task resume];

代理方法的实现遵守的协议是:NSURLSessionDownloadDelegate----代理方法中会把接收完成的数据,存放在临时文件中, 我们需要将文件调出来,放在我们希望的位置location

#pragma mark - NSURLSessionDownloadDelegate
/**
 *  开始写入数据
 *
 *  @param session
 *  @param downloadTask              当前下载任务
 *  @param bytesWritten              当前这次写入数据的大小
 *  @param totalBytesWritten         已经写入数据的大小
 *  @param totalBytesExpectedToWrite 预计写入数据的总大小
 */
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite
{
    self.progressView.progress = (float)1.0*totalBytesWritten / totalBytesExpectedToWrite ;
}
/**
 *  恢复和暂停
 *
 *  @param session
 *  @param downloadTask       当前下载任务
 *  @param fileOffset         恢复之后又从文件的什么地方开始进行写入
 *  @param expectedTotalBytes 文件剩余总大小
 */
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didResumeAtOffset:(int64_t)fileOffset expectedTotalBytes:(int64_t)expectedTotalBytes
{
 NSLog(@"%s",__func__);
}
/**
 *  下载完成时
 *
 *  @param session
 *  @param downloadTask 当前下载任务 (属性中有响应头)
 *  @param location     下载的位置
 */
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location
{
    NSURLResponse *response = downloadTask.response;
    
    NSString *toPath = [response.suggestedFilename cacheDir];
    NSLog(@"%@",toPath);
    
    NSFileManager *mgr = [NSFileManager defaultManager];
    
    // 将保存的临时文件,转移到,指定目录
    [mgr moveItemAtURL:location toURL:[NSURL fileURLWithPath:toPath] error:nil];

}

// 完成任务
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
 NSLog(@"%s",__func__);

}

NSURLSessionDownLoadTask--->断点下载的应用
NSURLSessionDownloadTask 为我们提供了新的断点下载的方法: cancelByProducingResumeData:void(^)(NSData *resumeData) 用于取消任务时,保存复原的数据,并记录当重新启动任务时,需要从何处继续加载 ---- 主要是用于暂停时数据的保留
dowmloadTaskWithResumeData:
必须在调用 resume才能开始任务
恢复下载时调用, 并且重新启动下载任务时 , 下载任务的起点已经发生了变化, 由传入的数据 开始启动任务---用于启动时,设置数据开始下载的起点

对于断点下载的优化------>对于每次启动程序, 我们并不需要重新下载, 只需要在本地沙盒中存储的数据--->继续下载就可以了

  1. 需要完成的就是, 发送请求的时候, 先读取本地数据, 并计算本地数据的大小, 然后在请求头设置时,加入请求数据开始范围
// 设置会话,并设置会话的代理, 以及回调队列
    NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
   
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_16.mp4"];
   
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    
   // 通过本地文件得到,下载了多少数据。 一旦重新启动任务,会从多少数据开始。 但是一旦重启,数据的响应头 中期望数据长度,将会改变,改变量为已将下好的长度,与总长度只差
    self.fileSize = [self fileSizeWith:Name];
    NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.fileSize];
  // 设置请求头内容
    [request setValue:range forHTTPHeaderField:@"Range"];
   // 创建下载 
    NSURLSessionDataTask *task = [session dataTaskWithRequest:request];
    self.task = task;
    [self.task resume];

2.创建方法,专门用于计算文件内容的大小

// 根据传入的文件名,来计算文件的大小
- (NSInteger)fileSizeWith:(NSString *)str;
{
    NSFileManager *mgr = [NSFileManager defaultManager];
    
    NSString *toPath = [str cacheDir];
// 文件中的相关参数
    NSDictionary *dict = [mgr attributesOfItemAtPath:toPath error:nil];
    
    return  [dict[NSFileSize] integerValue];
}

对于多次启动相同下载任务的优化(懒加载任务)

#define Name  @"minion_16.mp4"
@implementation ViewController
- (NSURLSessionDataTask *)task
{
    if (!_task) {
        NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
        
        NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_16.mp4"];
        
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
     
        self.fileSize = [self fileSizeWith:Name];
        NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.fileSize];
        
        [request setValue:range forHTTPHeaderField:@"Range"];
        
    NSURLSessionDataTask *task = [session dataTaskWithRequest:request];
        self.task = task;

    }
    return _task;
}

Tools 工具类:

为了能更方便的进行下载内容存储,所以我们给下载文件到沙盒路径提供更为简单的方法, 外界调用最方便的方法.
对NSString添加分类,创建实现方法

// 传入字符串,直接在沙盒Cache中生成路径
- (instancetype)cacheDir
{
    NSString *cache = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES).firstObject;
   return [cache stringByAppendingPathComponent:[self lastPathComponent]];
}
// 传入字符串,直接在沙盒Document中生成路径
- (instancetype)docDir
{

    NSString *doc = NSSearchPathForDirectoriesInDomains(NSDocumentationDirectory, NSUserDomainMask, YES).firstObject;
    return [doc stringByAppendingPathComponent:[self lastPathComponent]];

}
// 传入字符串,直接在沙盒Temp中生成路径
- (instancetype)temDir
{

    NSString *tem = NSTemporaryDirectory();
    return [tem stringByAppendingPathComponent:[self lastPathComponent]];

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

推荐阅读更多精彩内容