iOS开发网络篇—大文件的多线程断点下载

简介:

iOS开发网络篇—多线程断点下载

说明:本文介绍多线程断点下载。项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件。因为实现过程较为复杂,所以下面贴出完整的代码。

实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100M,那么就在沙盒中创建一个100M的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分)。

项目中用到的主要类如下:

完成的实现代码如下:

主控制器中的代码:

复制代码
 1 #import "YYViewController.h"  2 #import "YYFileMultiDownloader.h"  3  4 @interface YYViewController ()
 5 @property (nonatomic, strong) YYFileMultiDownloader *fileMultiDownloader;
 6 @end  7  8 @implementation YYViewController
 9 - (YYFileMultiDownloader *)fileMultiDownloader
10 {
11 if (!_fileMultiDownloader) {
12 _fileMultiDownloader = [[YYFileMultiDownloader alloc] init];
13 // 需要下载的文件远程URL 14 _fileMultiDownloader.url = @"http://192.168.1.200:8080/MJServer/resources/jre.zip";
15 // 文件保存到什么地方 16 NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
17 NSString *filepath = [caches stringByAppendingPathComponent:@"jre.zip"];
18 _fileMultiDownloader.destPath = filepath;
19  }
20 return _fileMultiDownloader;
21 }
22 23 - (void)viewDidLoad
24 {
25  [super viewDidLoad];
26 27 }
28 29 - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event 30 {
31  [self.fileMultiDownloader start];
32 }
33 34 @end
复制代码

自定义一个基类

YYFileDownloader.h文件

复制代码
 1 #import <Foundation/Foundation.h>
 2  3 @interface YYFileDownloader : NSObject
 4 {
 5  BOOL _downloading;
 6 }
 7 /**
 8  * 所需要下载文件的远程URL(连接服务器的路径)
 9 */ 10 @property (nonatomic, copy) NSString *url;
11 /**
12  * 文件的存储路径(文件下载到什么地方)
13 */ 14 @property (nonatomic, copy) NSString *destPath;
15 16 /**
17  * 是否正在下载(有没有在下载, 只有下载器内部才知道)
18 */ 19 @property (nonatomic, readonly, getter = isDownloading) BOOL downloading;
20 21 /**
22  * 用来监听下载进度
23 */ 24 @property (nonatomic, copy) void (^progressHandler)(double progress);
25 26 /**
27  * 开始(恢复)下载
28 */ 29 - (void)start;
30 31 /**
32  * 暂停下载
33 */ 34 - (void)pause;
35 @end
复制代码

YYFileDownloader.m文件

1 #import "YYFileDownloader.h" 2 3 @implementation YYFileDownloader
4 @end

下载器类继承自YYFileDownloader这个类

YYFileSingDownloader.h文件

复制代码
 1 #import "YYFileDownloader.h"  2  3 @interface YYFileSingleDownloader : YYFileDownloader
 4 /**
 5  * 开始的位置
 6 */  7 @property (nonatomic, assign) long long begin;
 8 /**
 9  * 结束的位置
10 */ 11 @property (nonatomic, assign) long long end; 
12 @end
复制代码

YYFileSingDownloader.m文件

复制代码
 1 #import "YYFileSingleDownloader.h"  2 @interface YYFileSingleDownloader() <NSURLConnectionDataDelegate>
 3 /**
 4  * 连接对象
 5 */  6 @property (nonatomic, strong) NSURLConnection *conn;
 7  8 /**
 9  * 写数据的文件句柄
 10 */  11 @property (nonatomic, strong) NSFileHandle *writeHandle;
 12 /**
 13  * 当前已下载数据的长度
 14 */  15 @property (nonatomic, assign) long long currentLength;
 16 @end  17  18 @implementation YYFileSingleDownloader
 19  20 - (NSFileHandle *)writeHandle
 21 {
 22 if (!_writeHandle) {
 23 _writeHandle = [NSFileHandle fileHandleForWritingAtPath:self.destPath];
 24  }
 25 return _writeHandle;
 26 }
 27  28 /**
 29  * 开始(恢复)下载
 30 */  31 - (void)start
 32 {
 33 NSURL *url = [NSURL URLWithString:self.url];
 34 // 默认就是GET请求  35 NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
 36 // 设置请求头信息  37 NSString *value = [NSString stringWithFormat:@"bytes=%lld-%lld", self.begin + self.currentLength, self.end];
 38 [request setValue:value forHTTPHeaderField:@"Range"];
 39 self.conn = [NSURLConnection connectionWithRequest:request delegate:self];
 40  41 _downloading = YES;
 42 }
 43  44 /**
 45  * 暂停下载
 46 */  47 - (void)pause
 48 {
 49  [self.conn cancel];
 50 self.conn = nil;
 51  52 _downloading = NO;
 53 }
 54  55  56 #pragma mark - NSURLConnectionDataDelegate 代理方法
 57 /**
 58  * 1. 当接受到服务器的响应(连通了服务器)就会调用
 59 */  60 - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
 61 {
 62  63 }
 64  65 /**
 66  * 2. 当接受到服务器的数据就会调用(可能会被调用多次, 每次调用只会传递部分数据)
 67 */  68 - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
 69 {
 70 // 移动到文件的尾部  71 [self.writeHandle seekToFileOffset:self.begin + self.currentLength];
 72 // 从当前移动的位置(文件尾部)开始写入数据  73  [self.writeHandle writeData:data];
 74  75 // 累加长度  76 self.currentLength += data.length;
 77  78 // 打印下载进度  79 double progress = (double)self.currentLength / (self.end - self.begin);
 80 if (self.progressHandler) {
 81  self.progressHandler(progress);
 82  }
 83 }
 84  85 /**
 86  * 3. 当服务器的数据接受完毕后就会调用
 87 */  88 - (void)connectionDidFinishLoading:(NSURLConnection *)connection
 89 {
 90 // 清空属性值  91 self.currentLength = 0;
 92  93 // 关闭连接(不再输入数据到文件中)  94  [self.writeHandle closeFile];
 95 self.writeHandle = nil;
 96 }
 97  98 /**
 99  * 请求错误(失败)的时候调用(请求超时\断网\没有网, 一般指客户端错误)
100 */ 101 - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
102 {
103 104 }
105 106 @end
复制代码

设计多线程下载器(利用HMFileMultiDownloader能开启多个线程同时下载一个文件)

一个多线程下载器只下载一个文件

YYFileMultiDownloader.h文件

1 #import "YYFileDownloader.h" 2 3 @interface YYFileMultiDownloader : YYFileDownloader
4 5 @end

YYFileMultiDownloader.m文件

复制代码
 1 #import "YYFileMultiDownloader.h"  2 #import "YYFileSingleDownloader.h"  3  4 #define YYMaxDownloadCount 4
 5  6 @interface YYFileMultiDownloader()
 7 @property (nonatomic, strong) NSMutableArray *singleDownloaders;
 8 @property (nonatomic, assign) long long totalLength;
 9 @end 10 11 @implementation YYFileMultiDownloader
12 13 - (void)getFilesize
14 {
15 NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:self.url]];
16 request.HTTPMethod = @"HEAD";
17 18 NSURLResponse *response = nil;
19 #warning 这里要用异步请求
20 [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:nil];
21 self.totalLength = response.expectedContentLength;
22 }
23 24 - (NSMutableArray *)singleDownloaders
25 {
26 if (!_singleDownloaders) {
27 _singleDownloaders = [NSMutableArray array];
28 29 // 获得文件大小 30  [self getFilesize];
31 32 // 每条路径的下载量 33 long long size = 0;
34 if (self.totalLength % YYMaxDownloadCount == 0) {
35 size = self.totalLength / YYMaxDownloadCount;
36 } else {
37 size = self.totalLength / YYMaxDownloadCount + 1;
38  }
39 40 // 创建N个下载器 41 for (int i = 0; i<YYMaxDownloadCount; i++) {
42 YYFileSingleDownloader *singleDownloader = [[YYFileSingleDownloader alloc] init];
43 singleDownloader.url = self.url;
44 singleDownloader.destPath = self.destPath;
45 singleDownloader.begin = i * size;
46 singleDownloader.end = singleDownloader.begin + size - 1;
47 singleDownloader.progressHandler = ^(double progress){
48 NSLog(@"%d --- %f", i, progress);
49  };
50  [_singleDownloaders addObject:singleDownloader];
51  }
52 53 // 创建一个跟服务器文件等大小的临时文件 54  [[NSFileManager defaultManager] createFileAtPath:self.destPath contents:nil attributes:nil];
55 56 // 让self.destPath文件的长度是self.totalLengt 57 NSFileHandle *handle = [NSFileHandle fileHandleForWritingAtPath:self.destPath];
58  [handle truncateFileAtOffset:self.totalLength];
59  }
60 return _singleDownloaders;
61 }
62 63 /**
64  * 开始(恢复)下载
65 */ 66 - (void)start
67 {
68  [self.singleDownloaders makeObjectsPerformSelector:@selector(start)];
69 70 _downloading = YES;
71 }
72 73 /**
74  * 暂停下载
75 */ 76 - (void)pause
77 {
78  [self.singleDownloaders makeObjectsPerformSelector:@selector(pause)];
79 _downloading = NO;
80 }
81 82 @end
复制代码

补充说明:如何获得将要下载的文件的大小?

目录
相关文章
|
2月前
|
机器学习/深度学习 存储 监控
内部文件审计:企业文件服务器审计对网络安全提升有哪些帮助?
企业文件服务器审计是保障信息安全、确保合规的关键措施。DataSecurity Plus 是由卓豪ManageEngine推出的审计工具,提供全面的文件访问监控、实时异常告警、用户行为分析及合规报告生成功能,助力企业防范数据泄露风险,满足GDPR、等保等多项合规要求,为企业的稳健发展保驾护航。
|
6月前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
219 23
|
消息中间件 存储 Serverless
函数计算产品使用问题之怎么访问网络附加存储(NAS)存储模型文件
函数计算产品作为一种事件驱动的全托管计算服务,让用户能够专注于业务逻辑的编写,而无需关心底层服务器的管理与运维。你可以有效地利用函数计算产品来支撑各类应用场景,从简单的数据处理到复杂的业务逻辑,实现快速、高效、低成本的云上部署与运维。以下是一些关于使用函数计算产品的合集和要点,帮助你更好地理解和应用这一服务。
|
9月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
java网络编程 UDP通信协议实现局域网内文件的发送和接收
java网络编程 UDP通信协议实现局域网内文件的发送和接收
151 1
java网络编程 UDP通信协议实现局域网内文件的发送和接收
|
12月前
|
API Android开发 iOS开发
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
|
API Windows
揭秘网络通信的魔法:Win32多线程技术如何让服务器化身超级英雄,同时与成千上万客户端对话!
【8月更文挑战第16天】在网络编程中,客户/服务器模型让客户端向服务器发送请求并接收响应。Win32 API支持在Windows上构建此类应用。首先要初始化网络环境并通过`socket`函数创建套接字。服务器需绑定地址和端口,使用`bind`和`listen`函数准备接收连接。对每个客户端调用`accept`函数并在新线程中处理。客户端则通过`connect`建立连接,双方可通过`send`和`recv`交换数据。多线程提升服务器处理能力,确保高效响应。
132 6
|
安全 测试技术 调度
iOS开发-多线程编程
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),保持对象不被释放;`weak`(弱引用),不保持对象,有助于避免循环引用;`assign`(赋值),适用于基本数据类型及非指针对象类型;`copy`(复制),复制对象而非引用,确保不变性。内存管理基于引用计数,利用自动引用计数(ARC)自动管理对象生命周期。此外,需注意避免循环引用,特别是在block中。最佳实践包括理解各策略、避免不必要的强引用、及时释放不再使用的对象、注意block中的内存管理,并使用工具进行内存分析。正确管理内存能显著提升应用质量。
106 2
|
存储 Ubuntu Linux
NFS服务部署全攻略:从零到一,轻松驾驭网络文件系统,让你的文件共享像飞一样畅快无阻!
【8月更文挑战第5天】NFS(网络文件系统)能让网络中的电脑无缝共享文件与目录。基于客户端-服务器模式,用户可像访问本地文件般透明操作远程文件。部署前需准备至少两台Linux机器:一台服务器,其余作客户端;确保已装NFS相关软件包且网络通畅。服务器端安装NFS服务与rpcbind,客户端安装nfs-utils。
697 4
|
安全 网络安全 数据安全/隐私保护
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享安卓与iOS开发中的线程管理比较
【8月更文挑战第30天】本文将探讨网络安全与信息安全的重要性,并分享关于网络安全漏洞、加密技术和安全意识的知识。我们将了解常见的网络攻击类型和防御策略,以及如何通过加密技术和提高安全意识来保护个人和组织的信息安全。

热门文章

最新文章