- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章iOS开发-实现大文件下载与断点下载思路由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
大文件下载 。
方案一:利用NSURLConnection和它的代理方法,及NSFileHandle(iOS9后不建议使用) 。
相关变量:
1
2
|
@property (nonatomic,strong) NSFileHandle *writeHandle;
@property (nonatomic,assign)
long
long
totalLength;
|
1>发送请求 。
1
2
3
4
5
|
// 创建一个请求
NSURL *url = [NSURL URLWithString:@
""
];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
// 使用NSURLConnection发起一个异步请求
[NSURLConnection connectionWithRequest:request delegate:self];
|
2>在代理方法中处理服务器返回的数据 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
/** 在接收到服务器的响应时调用下面这个代理方法
1.创建一个空文件
2.用一个句柄对象关联这个空文件,目的是方便在空文件后面写入数据
*/
- (
void
)connection:(NSURLConnection *)connection didReceiveResponse:(nonnull NSURLResponse *)response
{
// 创建文件路径
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject];
NSString *filePath = [caches stringByAppendingPathComponent:@
"videos.zip"
];
// 创建一个空的文件到沙盒中
NSFileManager *mgr = [NSFileManager defaultManager];
[mgr createFileAtPath:filePath contents:nil attributes:nil];
// 创建一个用来写数据的文件句柄
self.writeHandle = [NSFileHandle fileHandleForWritingAtPath:filePath];
// 获得文件的总大小
self.totalLength = response.expectedContentLength;
}
/** 在接收到服务器返回的文件数据时调用下面这个代理方法
利用句柄对象往文件的最后面追加数据
*/
- (
void
)connection:(NSURLConnection *)connection didReceiveData:(nonnull NSData *)data
{
// 移动到文件的最后面
[self.writeHandle seekToEndOfFile];
// 将数据写入沙盒
[self.writeHandle writeData:data];
}
/**
在所有数据接收完毕时,关闭句柄对象
*/
- (
void
)connectionDidFinishLoading:(NSURLConnection *)connection
{
// 关闭文件并清空
[self.writeHandle closeFile];
self.writeHandle = nil;
}
|
方案二:使用NSURLSession的NSURLSessionDownloadTask和NSFileManager 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
NSURLSession *session = [NSURLSession sharedSession];
NSURL *url = [NSURL URLWithString:@
""
];
// 可以用来下载大文件,数据将会存在沙盒里的tmp文件夹
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url completionHandler:^(NSURL * _Nullable location, NSURLResponse * _Nullable response, NSError * _Nullable error) {
// location :临时文件存放的路径(下载好的文件)
// 创建存储文件路径
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject];
// response.suggestedFilename:建议使用的文件名,一般跟服务器端的文件名一致
NSString *file = [caches stringByAppendingPathComponent:response.suggestedFilename];
/**将临时文件剪切或者复制到Caches文件夹
AtPath :剪切前的文件路径
toPath :剪切后的文件路径
*/
NSFileManager *mgr = [NSFileManager defaultManager];
[mgr moveItemAtPath:location.path toPath:file error:nil];
}];
[task resume];
|
方案三:使用NSURLSessionDownloadDelegate的代理方法和NSFileManger 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
|
- (
void
)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
// 创建一个下载任务并设置代理
NSURLSessionConfiguration *cfg = [NSURLSessionConfiguration defaultSessionConfiguration];
NSURLSession *session = [NSURLSession sessionWithConfiguration:cfg delegate:self delegateQueue:[NSOperationQueue mainQueue]];
NSURL *url = [NSURL URLWithString:@
""
];
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url];
[task resume];
}
#pragma mark -
/**
下载完毕后调用
参数:lication 临时文件的路径(下载好的文件)
*/
- (
void
)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location{
// 创建存储文件路径
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject];
// response.suggestedFilename:建议使用的文件名,一般跟服务器端的文件名一致
NSString *file = [caches stringByAppendingPathComponent:downloadTask.response.suggestedFilename];
/**将临时文件剪切或者复制到Caches文件夹
AtPath :剪切前的文件路径
toPath :剪切后的文件路径
*/
NSFileManager *mgr = [NSFileManager defaultManager];
[mgr moveItemAtPath:location.path toPath:file error:nil];
}
/**
每当下载完一部分时就会调用(可能会被调用多次)
参数:
bytesWritten 这次调用下载了多少
totalBytesWritten 累计写了多少长度到沙盒中了
totalBytesExpectedToWrite 文件总大小
*/
- (
void
)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didWriteData:(int64_t)bytesWritten
totalBytesWritten:(int64_t)totalBytesWritten
totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite{
// 这里可以做些显示进度等操作
}
/**
恢复下载时使用
*/
- (
void
)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didResumeAtOffset:(int64_t)fileOffset
expectedTotalBytes:(int64_t)expectedTotalBytes
{
// 用于断点续传
}
|
断点下载 。
方案一:
1>在方案一的基础上新增两个变量和按扭 。
1
2
|
@property (nonatomic,assign)
long
long
currentLength;
@property (nonatomic,strong) NSURLConnection *conn;
|
2>在接收到服务器返回数据的代理方法中添加如下代码 。
1
2
|
// 记录断点,累计文件长度
self.currentLength += data.length;
|
3>点击按钮开始(继续)或暂停下载 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
- (IBAction)download:(UIButton *)sender {
sender.selected = !sender.isSelected;
if
(sender.selected) {
// 继续(开始)下载
NSURL *url = [NSURL URLWithString:@
""
];
// ****关键点是使用NSMutableURLRequest,设置请求头Range
NSMutableURLRequest *mRequest = [NSMutableURLRequest requestWithURL:url];
NSString *range = [NSString stringWithFormat:@
"bytes=%lld-"
,self.currentLength];
[mRequest setValue:range forHTTPHeaderField:@
"Range"
];
// 下载
self.conn = [NSURLConnection connectionWithRequest:mRequest delegate:self];
}
else
{
[self.conn cancel];
self.conn = nil;
}
}
|
4>在接受到服务器响应执行的代理方法中第一行添加下面代码,防止重复创建空文件 。
1
|
if
(self.currentLength)
return
;
|
方案二:使用NSURLSessionDownloadDelegate的代理方法 。
所需变量 。
1
2
3
|
@property (nonatomic,strong) NSURLSession *session;
@property (nonatomic,strong) NSData *resumeData;
//包含了继续下载的开始位置和下载的url
@property (nonatomic,strong) NSURLSessionDownloadTask *task;
|
方法 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
|
// 懒加载session
- (NSURLSession *)session
{
if
(!_session) {
NSURLSessionConfiguration *cfg = [NSURLSessionConfiguration defaultSessionConfiguration];
self.session = [NSURLSession sessionWithConfiguration:cfg delegate:self delegateQueue:[NSOperationQueue mainQueue]];
}
return
_session;
}
- (IBAction)download:(UIButton *)sender {
sender.selected = !sender.isSelected;
if
(self.task == nil) {
// 开始(继续)下载
if
(self.resumeData) {
// 原先有数据则恢复
[self resume];
}
else
{
[self start];
// 原先没有数据则开始
}
}
else
{
// 暂停
[self pause];
}
}
// 从零开始
- (
void
)start{
NSURL *url = [NSURL URLWithString:@
""
];
self.task = [self.session downloadTaskWithURL:url];
[self.task resume];
}
// 暂停
- (
void
)pause{
__weak typeof(self) vc = self;
[self.task cancelByProducingResumeData:^(NSData * _Nullable resumeData) {
//resumeData : 包含了继续下载的开始位置和下载的url
vc.resumeData = resumeData;
vc.task = nil;
}];
}
// 恢复
- (
void
)resume{
// 传入上次暂停下载返回的数据,就可以回复下载
self.task = [self.session downloadTaskWithResumeData:self.resumeData];
// 开始任务
[self.task resume];
// 清空
self.resumeData = nil;
}
#pragma mark - NSURLSessionDownloadDelegate
/**
下载完毕后调用
参数:lication 临时文件的路径(下载好的文件)
*/
- (
void
)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location{
// 创建存储文件路径
NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject];
// response.suggestedFilename:建议使用的文件名,一般跟服务器端的文件名一致
NSString *file = [caches stringByAppendingPathComponent:downloadTask.response.suggestedFilename];
/**将临时文件剪切或者复制到Caches文件夹
AtPath :剪切前的文件路径
toPath :剪切后的文件路径
*/
NSFileManager *mgr = [NSFileManager defaultManager];
[mgr moveItemAtPath:location.path toPath:file error:nil];
}
/**
每当下载完一部分时就会调用(可能会被调用多次)
参数:
bytesWritten 这次调用下载了多少
totalBytesWritten 累计写了多少长度到沙盒中了
totalBytesExpectedToWrite 文件总大小
*/
- (
void
)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didWriteData:(int64_t)bytesWritten
totalBytesWritten:(int64_t)totalBytesWritten
totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite{
// 这里可以做些显示进度等操作
}
/**
恢复下载时使用
*/
- (
void
)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didResumeAtOffset:(int64_t)fileOffset
expectedTotalBytes:(int64_t)expectedTotalBytes
{
}
|
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我.
原文链接:http://www.cnblogs.com/jierism/p/6284304.html 。
最后此篇关于iOS开发-实现大文件下载与断点下载思路的文章就讲到这里了,如果你想了解更多关于iOS开发-实现大文件下载与断点下载思路的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
当我尝试通过我的 .exe 文件从 url 下载 .pdf 文件时出现以下错误。 The server committed a protocol violation. Section=Response
我是一家非营利组织的 G Suite 管理员,刚刚发现数据导出功能,这似乎是个人帐户的外卖。 导出文件已准备好,现在可以从 Google Cloud Platform Storage 中的存储桶下载。
导航 引言 总体思路 七牛云相关的配置文件 获取七牛云上传token 相关类定义 核心代码实现 获取七牛云图片下载链接 公开空
这不是后端编程问题。我只能修改标记或脚本(或文档本身)。我在这里问的原因是因为我对适当术语的所有搜索都不可避免地导致有关编程此功能的问题和解决方案。我不是试图通过编程来强制它;我必须找出此 PDF 行
您好,我已在 Google AdSense 中注册,我想使用适用于 iOS 的 SDK,但目前我找不到 SDK 下载链接。 我的申请已获批准。 任何人都知道如何下载这个sdk。 我使用这个链接来描述如
我需要为当前在 SourceForge 上的 github 项目提供二进制文件和文档。在那里,我可以为我需要的下载提供一个目录结构,因为我必须为大约 10 个不同的操作系统提供几个版本。 github
我从 Canvas 下载绘图时遇到问题。这是我的代码: function downloadCanvas(link, canvasId, filename) { link.href =
ASP.NET 项目 我将使用 Azure 进行存储。问题(要求): 在我的项目中,我让注册用户下载文件。但我不希望用户将此下载链接分享给未注册的人(例如:我给注册用户的下载链接只能在他们的计算机上下
我编写了一个servlet,用于检查http header ,但我不知道为什么当页面加载时,它会自动开始下载。 /* * To change this template, choose To
我正在尝试将下载添加到我的网络浏览器,但遇到的问题是获取您尝试下载的文件的名称。这是我的下载代码: engine.locationProperty().addListener(new ChangeLi
我正在尝试下载网站的 html: String encoding = "UTF-8"; HttpContext localContext = new BasicHttpContext();
我制作了一个带有“开始下载”按钮的框架,用于从网站下载 JAR。 问题是每当我点击开始下载按钮时,整个框架就会卡住,直到下载完成,然后就正常了。 我该如何解决这个问题? 这是单击按钮时执行的代码 p
我得到这段代码来实现一些东西,它可以帮助我从给定的 URL 下载文件。 -(void)URLSession:(NSURLSession *)session downloadTask:(NSURLSes
我正在尝试创建一个 Controller 来使用流方法下载和上传文件,在我的例子中,所有文件都作为 Blob 保存在数据库中。我阅读了 Jboss Netty 的文档,但我认为这不是我的最佳解决方案。
下载并保存文件 let destination: DownloadRequest.DownloadFileDestination = { _, _ in // var fileURL = sel
使用 htaccess 我基本上试图禁止访问该页面,即 http://example.com , 但它仍然允许人们下载文件,如果他们有直接链接即 http://example.com/hi.zip .
我正在寻求将脚本与我的控制面板集成,并且由于我是新手脚本编写者而遇到问题。我想做的是用 1 个脚本下载一个文件并解压它。 示例: wget http://example.com/example.tar
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
这个问题在这里已经有了答案: Top techniques to avoid 'data scraping' from a website database (14 个答案) 关闭 5 年前。 我有
这个问题在这里已经有了答案: Reading and parsing email from Gmail using C#, C++ or Python (6 个答案) 关闭 7 年前。 我只是想,是
我是一名优秀的程序员,十分优秀!