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

 //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 == ) {
//(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 != ) {
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 == ) {
//(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;
}

NSURLSession的文件下载的更多相关文章

  1. NSURLSession详解

    导语 现在NSURLConnection在开发中会使用的越来越少,iOS9已经将NSURLConnection废弃,现在最低版本一般适配iOS7,所以也可以使用. NSURLConnection相对于 ...

  2. 网络热恋之NSURLSession

    // // ViewController.m // NSURLSession代理简介 #import "ViewController.h" @interface ViewContr ...

  3. IOS 网络浅析-(十 NSURLSession下载简介)

    之前本来打算在写两个篇幅,但是在这片开写的时候觉得还是写一个比较好,有利于理解.NSURLSession下载是通过NSURLSession下载代理实现的,上一片也介绍了代理,之所以没有介绍下载是因为, ...

  4. iOS - NetRequest 网络数据请求

    1.网络请求 1.1 网络通讯三要素 1.IP 地址(主机名): 网络中设备的唯一标示.不易记忆,可以用主机名(域名). 1) IP V4: 0~255.0~255.0~255.0~255 ,共有 2 ...

  5. 文件下载(NSURLConnection/NSURLSession)

    最基本的网络文件下载(使用原生的网络请求) #pragma mark - 小文件下载 // 方法一: NSData dataWithContentsOfURL - (void)downloadFile ...

  6. [OC] NSURLSession

    有的程序员老了,还没听过NSURLSession 有的程序员还嫩,没用过NSURLConnection 有的程序员很单纯,他只知道AFN. NSURLConnection在iOS9被宣布弃用,NSUR ...

  7. iOS开发之Alamofire源码解析前奏--NSURLSession全家桶

    今天博客的主题不是Alamofire, 而是iOS网络编程中经常使用的NSURLSession.如果你想看权威的NSURLSession的东西,那么就得去苹果官方的开发中心去看了,虽然是英文的,但是结 ...

  8. [swift]NSURLSession

    一.简单说明 在iOS9.0之后,以前使用的NSURLConnection过期,苹果推荐使用NSURLSession来替换NSURLConnection完成网路请求相关操作. NSURLSession ...

  9. iOS网络2——NSURLSession使用详解

    原文在此 一.整体介绍 NSURLSession在2013年随着iOS7的发布一起面世,苹果对它的定位是作为NSURLConnection的替代者,然后逐步将NSURLConnection退出历史舞台 ...

随机推荐

  1. DevExpress的TreeList实现显示本地文件目录并自定义右键实现删除与重命名文件

    场景 使用DevExpress的TreeList显示本磁盘下文件目录并在树节点上右键实现删除与添加文件. 效果 自定义右键效果 实现 首先在包含Treelist的窗体的load方法中对treelist ...

  2. 【C#】学习笔记(4) 值类型和引用类型相关(Null相关)

    Reference and Value Types Value Types(值类型): struct(结构体) 独立的实例或者是拷贝 值的改变不会影响其它拷贝 值就是它所代表的信息 没有引用,所以不可 ...

  3. iOS中计算字符串NSString的高度

    根据固定宽度计算字符串高度: NSString *info = @"但是公司的高度是广东省公司的广东省高速度来开个大帅哥多撒谎个爱好就跟他说噶三公司噶是的刚好是我哥如果黑暗如果坏都干撒降低公 ...

  4. HTTP 简述

    HTTP 简介: 1.Hyper Text Transfer Protocol(超文本传输协议),主要用于 Web 浏览器和 Web 服务器之间的通信 2.它基于 TCP/IP 通信协议来传输数据 3 ...

  5. CodeForces - 763A(并查集/思维)

    题意 https://vjudge.net/problem/CodeForces-763A 一棵无根树中各个节点被染上了一种颜色c[i] 现在让你选择一个点作为根节点,使得这个根节点的所有儿子满足以该 ...

  6. 8. java 面向对象

    一.面向对象特征 1. 封装 方法就是一种封装 关键字private也是一种封装 封装就是讲一些逻辑细节信息隐藏起来,对于外界不可见:外界只需调用我即可: 一旦使用了private进行修饰,那么本类当 ...

  7. LeetCode 5129. 下降路径最小和 II Minimum Falling Path Sum II

    地址 https://leetcode-cn.com/contest/biweekly-contest-15/problems/minimum-falling-path-sum-ii/ 题目描述给你一 ...

  8. WPF 精修篇 数据绑定 更新通知

    原文:WPF 精修篇 数据绑定 更新通知 开始更新一点有意思的了 首先 数据绑定  其中之一 Element 绑定 看例子 <Window x:Class="WpfApplicatio ...

  9. php array()和[]

    比较数组 array() 和 [] 执行结果:(其中之一) array() : 执行时间在0.015-0.55之间 [] : 执行时间在0.015-0.35之间 结论: []执行时间更少更稳定

  10. Redis 笔记整理:回收策略与 LRU 算法

    Redis的回收策略 noeviction:返回错误当内存限制达到并且客户端尝试执行会让更多内存被使用的命令(大部分的写入指令,但DEL和几个例外) allkeys-lru: 尝试回收最少使用的键(L ...