欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2023-12-19 09:22:04
说明:本文介绍多线程断点下载。项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件。因为实现过程较为复杂,所以下面贴出完整的代码。 实现思路:下载开始,创建...

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

实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100m,那么就在沙盒中创建一个100m的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分)。
iOS开发网络篇—实现大文件的多线程断点下载
项目中用到的主要类如下:
iOS开发网络篇—实现大文件的多线程断点下载

完成的实现代码如下:

主控制器中的代码:

#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

补充说明:如何获得将要下载的文件的大小?iOS开发网络篇—实现大文件的多线程断点下载
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

上一篇:

下一篇: