• NSURLSession的文件下载


    小文件的下载,代码示例:

     //NSURLSession的下载
        [[[NSURLSession sharedSession] dataTaskWithURL:[NSURL URLWithString:@"http://gss0.baidu.com/-fo3dSag_xI4khGko9WTAnF6hhy/lvpics/w=600/sign=1350023d79899e51788e391472a5d990/b21bb051f819861810d03e4448ed2e738ad4e65f.jpg"] completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
            //回到主线程刷新界面
            dispatch_async(dispatch_get_main_queue(), ^{
                self.imageView.image = [UIImage imageWithData:data];
            });
            
        }] resume] ;

    大文件的下载,代码示例:

    #import "ViewController.h"
    
    @interface ViewController ()<NSURLSessionDataDelegate>
    
    @property (weak, nonatomic) IBOutlet UIImageView *imageView;
    
    @property (nonatomic,strong) NSMutableData * fileData ;
    @property (nonatomic,assign) NSInteger  totalSize ;
    @property (nonatomic,assign) NSInteger  currentSize ;
    @property (weak, nonatomic) IBOutlet UIProgressView *progressView;
    
    
    @end
    
    @implementation ViewController- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
       
        //1 确定资源路径
        NSURL *url = [NSURL URLWithString:@"http://meiye-mbs.oss-cn-shenzhen.aliyuncs.com/mbsFiles/0e3d0e4a0d5d4da5963e9e7617e8de101565841097849.mp4"];
        //2 创建请求对象
        NSURLRequest *request = [NSURLRequest requestWithURL:url];
        //3 创建会话对象:线程不传,默认在子线程中处理
        NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
        //4 创建下载请求Task
        NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];
        //5 发送请求
        [dataTask resume];
    }
    
    #pragma mark - 代理方法
    //1 接收到响应的时候调用
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
        
        //得到本次请求的文件数据大小
        self.totalSize = response.expectedContentLength;
        
        //告诉系统应该接收数据
        completionHandler(NSURLSessionResponseAllow);
    }
    
    //2 接收到服务器返回数据的时候调用,可能会调用多次
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
        //拼接数据
        [self.fileData appendData:data];
        
        //计算文件的下载进度并显示= 已经下载的数据/文件的总大小
        self.currentSize += data.length;
        CGFloat progress = 1.0 * self.currentSize / self.totalSize;
        self.progressView.progress = progress;
        NSLog(@"%f", progress);
    }
    
    //3 下载完成或者失败的时候调用
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    
        //得到文件的名称:得到请求的响应头信息,获取响应头信息中推荐的文件名称
        NSString *fileName = [task.response suggestedFilename];
        //拼接文件的存储路径(沙盒路径Cache + 文件名)
        NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        //下载的路径
        NSString *fullPath = [cache stringByAppendingPathComponent:fileName];
    
        //下载好的数据写入到磁盘
        [self.fileData writeToFile:fullPath atomically:YES];
        NSLog(@"filePath = %@", fullPath);
    }
    
    
    - (NSMutableData *)fileData {
        if (!_fileData) {
            _fileData = [NSMutableData data];
        }
        return _fileData;
    }

    如果按照上面那段代码的话,可以实现功能,但是有一个问题,就是 内存会飚升。为了处理这个问题,直接把拿到的数据,就写入沙盒,

    逻辑如下:

    //文件句柄(指针)NSFileHandle
    /**
     特点:在写数据的时候边写数据边移动位置
     使用步骤:
     (1)创建空的文件
     (2)创建文件句柄指针指向文件
     (3)当接收到数据的时候,使用该句柄来写数据
     (4)当所有的数据写完,应该关闭句柄指针
     */

    修改如下:

    #import "ViewController.h"
    
    @interface ViewController ()<NSURLSessionDataDelegate>
    
    @property (nonatomic,assign) NSInteger  totalSize ;
    @property (nonatomic,assign) NSInteger  currentSize ;
    @property (weak, nonatomic) IBOutlet UIProgressView *progressView;
    @property (nonatomic,strong) NSFileHandle * handle ;
    
    
    @end
    
    @implementation ViewController
    
    
    - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
       
        //1 确定资源路径
        NSURL *url = [NSURL URLWithString:@"http://meiye-mbs.oss-cn-shenzhen.aliyuncs.com/mbsFiles/0e3d0e4a0d5d4da5963e9e7617e8de101565841097849.mp4"];
        //2 创建请求对象
        NSURLRequest *request = [NSURLRequest requestWithURL:url];
        //3 创建会话对象:线程不传,默认在子线程中处理
        NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
        //4 创建下载请求Task
        NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];
        //5 发送请求
        [dataTask resume];
    }
    
    #pragma mark - 代理方法
    //1 接收到响应的时候调用
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
        
        //得到文件的名称:得到请求的响应头信息,获取响应头信息中推荐的文件名称
        NSString *fileName = [response suggestedFilename];
        //拼接文件的存储路径(沙盒路径Cache + 文件名)
        NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        //下载的路径
        NSString *fullPath = [cache stringByAppendingPathComponent:fileName];
        
        //(1)创建空的文件
        [[NSFileManager defaultManager] createFileAtPath:fullPath contents:nil attributes:nil];
        //(2)创建文件句柄指针指向文件
         self.handle = [NSFileHandle fileHandleForWritingAtPath:fullPath];
        
        //得到本次请求的文件数据大小
        self.totalSize = response.expectedContentLength;
        
        //告诉系统应该接收数据
        completionHandler(NSURLSessionResponseAllow);
    }
    
    //2 接收到服务器返回数据的时候调用,可能会调用多次
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
        [self.handle writeData:data];
        
        //计算文件的下载进度并显示= 已经下载的数据/文件的总大小
        self.currentSize += data.length;
        CGFloat progress = 1.0 * self.currentSize / self.totalSize;
        self.progressView.progress = progress;
        NSLog(@"%f", progress);
    }
    
    //3 下载完成或者失败的时候调用
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
        
        //(4)当所有的数据写完,应该关闭句柄指针
        [self.handle closeFile];
    }

     按照上面的代码可以解决内存的问题,但是,下完后,会发现,文件打不开,因为文件被损坏了,而且查看文件的大小会发现,文件的内存变小了。

    出现这个问题的原因是:

    (1)重新下载的时候,会重新创建一个空的文件,把之前下载的内容覆盖了。

    (2)当取消下载的时候,文件句柄会默认从文件的开头开始存储,这样就会覆盖之前的内容。

    解决思路:

    (1)判断是否是第一次下载,如果是第一次下载就创建一个空的文件夹

    (2)让文件句柄指针指向文件的末尾

    代码修改如下:

    #import "ViewController.h"
    
    @interface ViewController ()<NSURLSessionDataDelegate>
    
    @property (nonatomic,strong) NSURLSessionDataTask *dataTask;
    
    @property (nonatomic,assign) NSInteger  totalSize ;
    @property (nonatomic,assign) NSInteger  currentSize ;
    @property (weak, nonatomic) IBOutlet UIProgressView *progressView;
    @property (nonatomic,strong) NSFileHandle * handle ;
    
    
    @end
    
    @implementation ViewController
    
    - (IBAction)startAction:(id)sender {
        [self.dataTask resume];
    }
    
    - (IBAction)suspendAction:(id)sender {
        [self.dataTask suspend];
    }
    
    - (IBAction)cancleAction:(id)sender {
        [self.dataTask cancel];
        self.dataTask = nil;
    }
    
    - (IBAction)resumeAction:(id)sender {
        [self.dataTask resume];
    }
    
    
    #pragma mark - 代理方法
    //1 接收到响应的时候调用
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
        
        //得到文件的名称:得到请求的响应头信息,获取响应头信息中推荐的文件名称
        NSString *fileName = [response suggestedFilename];
        //拼接文件的存储路径(沙盒路径Cache + 文件名)
        NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
        //下载的路径
        NSString *fullPath = [cache stringByAppendingPathComponent:fileName];
        
        //判断是否是第一次发送请求下载,只有在第一次请求下载的时候才需要创建文件
        if (self.currentSize == 0) {
            //(1)创建空的文件
            [[NSFileManager defaultManager] createFileAtPath:fullPath contents:nil attributes:nil];
        }
        
        //(2)创建文件句柄指针指向文件:默认指向文件的开头
         self.handle = [NSFileHandle fileHandleForWritingAtPath:fullPath];
        
        //(3)移动文件句柄指针指向文件的末尾
        [self.handle seekToEndOfFile];
        
        //得到本次请求的文件数据大小
        self.totalSize = response.expectedContentLength + self.currentSize;
        
        //告诉系统应该接收数据
        completionHandler(NSURLSessionResponseAllow);
    }
    
    //2 接收到服务器返回数据的时候调用,可能会调用多次
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
        [self.handle writeData:data];
        
        //计算文件的下载进度并显示= 已经下载的数据/文件的总大小
        self.currentSize += data.length;
        CGFloat progress = 1.0 * self.currentSize / self.totalSize;
        self.progressView.progress = progress;
        NSLog(@"%f", progress);
    }
    
    //3 下载完成或者失败的时候调用
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
        
        //(4)当所有的数据写完,应该关闭句柄指针
        [self.handle closeFile];
    }
    
    #pragma mark - 懒加载
    - (NSURLSessionDataTask *)dataTask {
        if (!_dataTask) {
            //1 确定资源路径
            NSURL *url = [NSURL URLWithString:@"http://meiye-mbs.oss-cn-shenzhen.aliyuncs.com/mbsFiles/0e3d0e4a0d5d4da5963e9e7617e8de101565841097849.mp4"];
            //2 创建请求对象
            NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
            
            // 设置请求头信息(告诉请求对象只下载某一部分数据)Range
            /**
             Range:bytes=0-100
                   bytes=-100  文件从-100开始
                   bytes=400-1000
                   bytes=400-  400个字节的位置开始一直到结束
             注意:不能有空格
             */
            NSString *header = [NSString stringWithFormat:@"bytes=%zd-",self.currentSize];
            [request setValue:header forHTTPHeaderField:@"Range"];
            
            //3 创建会话对象:线程不传,默认在子线程中处理
            NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
            //4 创建下载请求Task
            _dataTask = [session dataTaskWithRequest:request];
        }
        return _dataTask;
    }

    最后,总结文件下载的思路:

    断点续传的思路:

    启动的时候,判断该路径下的文件夹,是否有数据,如果有数据的话,就在原先的基础上继续下载。

    代码如下:

    #import "ViewController.h"
    
    #define kFullPath [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"ceshi.mp4"]
    #define kSizeFullPath [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"ceshi.ceshi"]
    
    @interface ViewController ()<NSURLSessionDataDelegate>
    
    @property (nonatomic,strong) NSURLSessionDataTask *dataTask;
    
    @property (nonatomic,assign) NSInteger  totalSize ;
    @property (nonatomic,assign) NSInteger  currentSize ;
    @property (weak, nonatomic) IBOutlet UIProgressView *progressView;
    @property (nonatomic,strong) NSFileHandle * handle ;
    
    
    @end
    
    @implementation ViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        
        //拿到之前已经下载的文件数据大小 self.currentSize = 沙盒中文件的大小
        NSDictionary *fileInfo = [[NSFileManager defaultManager] attributesOfItemAtPath:kFullPath error:nil];
        NSLog(@"---%@---",fileInfo);
        self.currentSize = [fileInfo fileSize];
        
        //处理进度信息 == 已经下载的大小/文件的总大小
        NSData *sizeData = [NSData dataWithContentsOfFile:kSizeFullPath];
        self.totalSize = [[[NSString alloc] initWithData:sizeData encoding:NSUTF8StringEncoding] integerValue];
        if (self.totalSize != 0) {
            NSLog(@"%f", 1.0 * self.currentSize / self.totalSize);
            self.progressView.progress = 1.0 * self.currentSize / self.totalSize;
        }
    }
    
    - (IBAction)startAction:(id)sender {
        [self.dataTask resume];
    }
    
    - (IBAction)suspendAction:(id)sender {
        [self.dataTask suspend];
    }
    
    - (IBAction)cancleAction:(id)sender {
        [self.dataTask cancel];
        self.dataTask = nil;
    }
    
    - (IBAction)resumeAction:(id)sender {
        [self.dataTask resume];
    }
    
    
    #pragma mark - 代理方法
    //1 接收到响应的时候调用
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
        
        
        //判断是否是第一次发送请求下载,只有在第一次请求下载的时候才需要创建文件
        if (self.currentSize == 0) {
            //(1)创建空的文件
            [[NSFileManager defaultManager] createFileAtPath:kFullPath contents:nil attributes:nil];
        }
        
        //(2)创建文件句柄指针指向文件:默认指向文件的开头
         self.handle = [NSFileHandle fileHandleForWritingAtPath:kFullPath];
        
        //(3)移动文件句柄指针指向文件的末尾
        [self.handle seekToEndOfFile];
        
        //得到本次请求的文件数据大小
        self.totalSize = response.expectedContentLength + self.currentSize;
        
        //把文件的总大小信息保存起来
        NSData *sizeData = [[NSString stringWithFormat:@"%zd", self.totalSize] dataUsingEncoding:NSUTF8StringEncoding];
        [sizeData writeToFile:kSizeFullPath atomically:YES];
        
        //告诉系统应该接收数据
        completionHandler(NSURLSessionResponseAllow);
    }
    
    //2 接收到服务器返回数据的时候调用,可能会调用多次
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
        [self.handle writeData:data];
        
        //计算文件的下载进度并显示= 已经下载的数据/文件的总大小
        self.currentSize += data.length;
        CGFloat progress = 1.0 * self.currentSize / self.totalSize;
        self.progressView.progress = progress;
        NSLog(@"%f", progress);
    }
    
    //3 下载完成或者失败的时候调用
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
        
        //(4)当所有的数据写完,应该关闭句柄指针
        [self.handle closeFile];
    }
    
    #pragma mark - 懒加载
    - (NSURLSessionDataTask *)dataTask {
        if (!_dataTask) {
            //1 确定资源路径
            NSURL *url = [NSURL URLWithString:@"http://meiye-mbs.oss-cn-shenzhen.aliyuncs.com/mbsFiles/0e3d0e4a0d5d4da5963e9e7617e8de101565841097849.mp4"];
            //2 创建请求对象
            NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
            
            // 设置请求头信息(告诉请求对象只下载某一部分数据)Range
            /**
             Range:bytes=0-100
                   bytes=-100  文件从-100开始
                   bytes=400-1000
                   bytes=400-  400个字节的位置开始一直到结束
             注意:不能有空格
             */
            NSString *header = [NSString stringWithFormat:@"bytes=%zd-",self.currentSize];
            [request setValue:header forHTTPHeaderField:@"Range"];
            
            //3 创建会话对象:线程不传,默认在子线程中处理
            NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
            //4 创建下载请求Task
            _dataTask = [session dataTaskWithRequest:request];
        }
        return _dataTask;
    }
  • 相关阅读:
    SAP MM VL32N和MIGO对内向交货单做收货,都会更新其'总体货物移动状态'
    Several mentality of SAP project customers in private enterprises
    GIT·AccessToken的基础使用
    MSSQL·按照某个字段重复删除旧的一条数据
    MSSQL·WHERE INT列=''时的结果集
    技能Get·手动更新HP笔记本BIOS过程记录
    GIT·.NetFramework MVC项目默认的.gitignore文件备份
    .Net·发布过程中报:在应用程序之外使用注册为allowDefinition='MachineTOApplication'的节是错误...
    知识总结·多系统数据同步API调用设计原则
    MSSQL·查看表锁进程及杀死进程的脚本
  • 原文地址:https://www.cnblogs.com/lyz0925/p/11580252.html
Copyright © 2020-2023  润新知