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

简介: iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载。项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件。因为实现过程较为复杂,所以下面贴出完整的代码。 实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100M,那么就在沙盒中创建一个100M的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分)。

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

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

目录
相关文章
|
6月前
|
网络协议 算法 Java
|
6月前
|
安全 Java 数据处理
Python网络编程基础(Socket编程)多线程/多进程服务器编程
【4月更文挑战第11天】在网络编程中,随着客户端数量的增加,服务器的处理能力成为了一个重要的考量因素。为了处理多个客户端的并发请求,我们通常需要采用多线程或多进程的方式。在本章中,我们将探讨多线程/多进程服务器编程的概念,并通过一个多线程服务器的示例来演示其实现。
|
6月前
|
Java
网络 I/O:单 Selector 多线程(单线程模型)
网络 I/O:单 Selector 多线程(单线程模型)
|
6月前
|
Java
Socket网络编程练习题五:客户端多用户上传文件(多线程版)并使用线程池管理线程
Socket网络编程练习题五:客户端多用户上传文件(多线程版)并使用线程池管理线程
|
6月前
|
网络协议 Linux C++
Linux C/C++ 开发(学习笔记十一 ):TCP服务器(并发网络网络编程 一请求一线程)
Linux C/C++ 开发(学习笔记十一 ):TCP服务器(并发网络网络编程 一请求一线程)
132 0
|
1月前
|
Java 应用服务中间件
面对海量网络请求,Tomcat线程池如何进行扩展?
【10月更文挑战第4天】本文详细探讨了Tomcat线程池相较于标准Java实用工具包(JUC)线程池的关键改进。首先,Tomcat线程池在启动时即预先创建全部核心线程,以应对启动初期的高并发请求。其次,通过重写阻塞队列的入队逻辑,Tomcat能够在任务数超过当前线程数但未达最大线程数时,及时创建非核心线程,而非等到队列满才行动。此外,Tomcat还引入了在拒绝策略触发后重新尝试入队的机制,以提高吞吐量。这些优化使得Tomcat线程池更适应IO密集型任务,有效提升了性能。
面对海量网络请求,Tomcat线程池如何进行扩展?
|
1月前
|
Java Linux
【网络】高并发场景处理:线程池和IO多路复用
【网络】高并发场景处理:线程池和IO多路复用
45 2
|
2月前
|
网络协议 C语言
C语言 网络编程(十四)并发的TCP服务端-以线程完成功能
这段代码实现了一个基于TCP协议的多线程服务器和客户端程序,服务器端通过为每个客户端创建独立的线程来处理并发请求,解决了粘包问题并支持不定长数据传输。服务器监听在IP地址`172.17.140.183`的`8080`端口上,接收客户端发来的数据,并将接收到的消息添加“-回传”后返回给客户端。客户端则可以循环输入并发送数据,同时接收服务器回传的信息。当输入“exit”时,客户端会结束与服务器的通信并关闭连接。
|
2月前
|
C语言
C语言 网络编程(九)并发的UDP服务端 以线程完成功能
这是一个基于UDP协议的客户端和服务端程序,其中服务端采用多线程并发处理客户端请求。客户端通过UDP向服务端发送登录请求,并根据登录结果与服务端的新子线程进行后续交互。服务端在主线程中接收客户端请求并创建新线程处理登录验证及后续通信,子线程创建新的套接字并与客户端进行数据交换。该程序展示了如何利用线程和UDP实现简单的并发服务器架构。
|
3月前
|
API Windows
揭秘网络通信的魔法:Win32多线程技术如何让服务器化身超级英雄,同时与成千上万客户端对话!
【8月更文挑战第16天】在网络编程中,客户/服务器模型让客户端向服务器发送请求并接收响应。Win32 API支持在Windows上构建此类应用。首先要初始化网络环境并通过`socket`函数创建套接字。服务器需绑定地址和端口,使用`bind`和`listen`函数准备接收连接。对每个客户端调用`accept`函数并在新线程中处理。客户端则通过`connect`建立连接,双方可通过`send`和`recv`交换数据。多线程提升服务器处理能力,确保高效响应。
54 6