小文件的下载,代码示例:
//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; }