iOS开发网络篇—实现大文件的多线程断点下载
程序员文章站
2023-12-19 09:22:04
说明:本文介绍多线程断点下载。项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件。因为实现过程较为复杂,所以下面贴出完整的代码。
实现思路:下载开始,创建...
说明:本文介绍多线程断点下载。项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件。因为实现过程较为复杂,所以下面贴出完整的代码。
实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100m,那么就在沙盒中创建一个100m的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分)。
项目中用到的主要类如下:
完成的实现代码如下:
主控制器中的代码:
#import "yyviewcontroller.h" #import "yyfilemultidownloader.h" @interface yyviewcontroller () @property (nonatomic, strong) yyfilemultidownloader *filemultidownloader; @end @implementation yyviewcontroller - (yyfilemultidownloader *)filemultidownloader { if (!_filemultidownloader) { _filemultidownloader = [[yyfilemultidownloader alloc] init]; // 需要下载的文件远程url _filemultidownloader.url = @"http://192.168.1.200:8080/mjserver/resources/jre.zip"; // 文件保存到什么地方 nsstring *caches = [nssearchpathfordirectoriesindomains(nscachesdirectory, nsuserdomainmask, yes) lastobject]; nsstring *filepath = [caches stringbyappendingpathcomponent:@"jre.zip"]; _filemultidownloader.destpath = filepath; } return _filemultidownloader; } - (void)viewdidload { [super viewdidload]; } - (void)touchesbegan:(nsset *)touches withevent:(uievent *)event { [self.filemultidownloader start]; } @end
自定义一个基类
yyfiledownloader.h文件
#import <foundation/foundation.h> @interface yyfiledownloader : nsobject { bool _downloading; } /** * 所需要下载文件的远程url(连接服务器的路径) */ @property (nonatomic, copy) nsstring *url; /** * 文件的存储路径(文件下载到什么地方) */ @property (nonatomic, copy) nsstring *destpath; /** * 是否正在下载(有没有在下载, 只有下载器内部才知道) */ @property (nonatomic, readonly, getter = isdownloading) bool downloading; /** * 用来监听下载进度 */ @property (nonatomic, copy) void (^progresshandler)(double progress); /** * 开始(恢复)下载 */ - (void)start; /** * 暂停下载 */ - (void)pause; @end
yyfiledownloader.m文件
#import "yyfiledownloader.h" @implementation yyfiledownloader @end 下载器类继承自yyfiledownloader这个类 yyfilesingdownloader.h文件 #import "yyfiledownloader.h" @interface yyfilesingledownloader : yyfiledownloader /** * 开始的位置 */ @property (nonatomic, assign) long long begin; /** * 结束的位置 */ @property (nonatomic, assign) long long end; @end yyfilesingdownloader.m文件 #import "yyfilesingledownloader.h" @interface yyfilesingledownloader() <nsurlconnectiondatadelegate> /** * 连接对象 */ @property (nonatomic, strong) nsurlconnection *conn; /** * 写数据的文件句柄 */ @property (nonatomic, strong) nsfilehandle *writehandle; /** * 当前已下载数据的长度 */ @property (nonatomic, assign) long long currentlength; @end @implementation yyfilesingledownloader - (nsfilehandle *)writehandle { if (!_writehandle) { _writehandle = [nsfilehandle filehandleforwritingatpath:self.destpath]; } return _writehandle; } /** * 开始(恢复)下载 */ - (void)start { nsurl *url = [nsurl urlwithstring:self.url]; // 默认就是get请求 nsmutableurlrequest *request = [nsmutableurlrequest requestwithurl:url]; // 设置请求头信息 nsstring *value = [nsstring stringwithformat:@"bytes=%lld-%lld", self.begin + self.currentlength, self.end]; [request setvalue:value forhttpheaderfield:@"range"]; self.conn = [nsurlconnection connectionwithrequest:request delegate:self]; _downloading = yes; } /** * 暂停下载 */ - (void)pause { [self.conn cancel]; self.conn = nil; _downloading = no; } #pragma mark - nsurlconnectiondatadelegate 代理方法 /** * 1. 当接受到服务器的响应(连通了服务器)就会调用 */ - (void)connection:(nsurlconnection *)connection didreceiveresponse:(nsurlresponse *)response { } /** * 2. 当接受到服务器的数据就会调用(可能会被调用多次, 每次调用只会传递部分数据) */ - (void)connection:(nsurlconnection *)connection didreceivedata:(nsdata *)data { // 移动到文件的尾部 [self.writehandle seektofileoffset:self.begin + self.currentlength]; // 从当前移动的位置(文件尾部)开始写入数据 [self.writehandle writedata:data]; // 累加长度 self.currentlength += data.length; // 打印下载进度 double progress = (double)self.currentlength / (self.end - self.begin); if (self.progresshandler) { self.progresshandler(progress); } } /** * 3. 当服务器的数据接受完毕后就会调用 */ - (void)connectiondidfinishloading:(nsurlconnection *)connection { // 清空属性值 self.currentlength = 0; // 关闭连接(不再输入数据到文件中) [self.writehandle closefile]; self.writehandle = nil; } /** * 请求错误(失败)的时候调用(请求超时\断网\没有网, 一般指客户端错误) */ - (void)connection:(nsurlconnection *)connection didfailwitherror:(nserror *)error { } @end
设计多线程下载器(利用hmfilemultidownloader能开启多个线程同时下载一个文件)
一个多线程下载器只下载一个文件
yyfilemultidownloader.h文件
#import "yyfiledownloader.h" @interface yyfilemultidownloader : yyfiledownloader @end
yyfilemultidownloader.m文件
#import "yyfilemultidownloader.h" #import "yyfilesingledownloader.h" #define yymaxdownloadcount 4 @interface yyfilemultidownloader() @property (nonatomic, strong) nsmutablearray *singledownloaders; @property (nonatomic, assign) long long totallength; @end @implementation yyfilemultidownloader - (void)getfilesize { nsmutableurlrequest *request = [nsmutableurlrequest requestwithurl:[nsurl urlwithstring:self.url]]; request.httpmethod = @"head"; nsurlresponse *response = nil; #warning 这里要用异步请求 [nsurlconnection sendsynchronousrequest:request returningresponse:&response error:nil]; self.totallength = response.expectedcontentlength; } - (nsmutablearray *)singledownloaders { if (!_singledownloaders) { _singledownloaders = [nsmutablearray array]; // 获得文件大小 [self getfilesize]; // 每条路径的下载量 long long size = 0; if (self.totallength % yymaxdownloadcount == 0) { size = self.totallength / yymaxdownloadcount; } else { size = self.totallength / yymaxdownloadcount + 1; } // 创建n个下载器 for (int i = 0; i<yymaxdownloadcount; i++) { yyfilesingledownloader *singledownloader = [[yyfilesingledownloader alloc] init]; singledownloader.url = self.url; singledownloader.destpath = self.destpath; singledownloader.begin = i * size; singledownloader.end = singledownloader.begin + size - 1; singledownloader.progresshandler = ^(double progress){ nslog(@"%d --- %f", i, progress); }; [_singledownloaders addobject:singledownloader]; } // 创建一个跟服务器文件等大小的临时文件 [[nsfilemanager defaultmanager] createfileatpath:self.destpath contents:nil attributes:nil]; // 让self.destpath文件的长度是self.totallengt nsfilehandle *handle = [nsfilehandle filehandleforwritingatpath:self.destpath]; [handle truncatefileatoffset:self.totallength]; } return _singledownloaders; } /** * 开始(恢复)下载 */ - (void)start { [self.singledownloaders makeobjectsperformselector:@selector(start)]; _downloading = yes; } /** * 暂停下载 */ - (void)pause { [self.singledownloaders makeobjectsperformselector:@selector(pause)]; _downloading = no; } @end
补充说明:如何获得将要下载的文件的大小?
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。