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

 更新时间:2016年11月03日 10:53:50   作者:文顶顶  
iOS开发中经常会用到文件的下载功能,这篇文章主要介绍了iOS开发网络篇—实现大文件的多线程断点下载,今天咱们来分享一下思路。

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

实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为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

补充说明:如何获得将要下载的文件的大小?
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • iOS中日志同步获取NSLog重定向以及其他详解

    iOS中日志同步获取NSLog重定向以及其他详解

    在Objective-c开发程序的时候,有专门的日志操作类NSLog,它将指定的输出,输出到(stderr),我们可以利用Xcode的日志输出窗口,下面这篇文章主要给大家介绍了关于iOS中日志同步获取NSLog重定向以及其他的相关资料,需要的朋友可以参考下。
    2017-12-12
  • xcode8提交ipa失败无法构建版本问题的解决方案

    xcode8提交ipa失败无法构建版本问题的解决方案

    xcode升级到xcode8后发现构建不了新的版本。怎么解决呢?下面小编给大家带来了xcode8提交ipa失败无法构建版本问题的解决方案,非常不错,一起看看吧
    2016-10-10
  • iOS实现波浪效果

    iOS实现波浪效果

    这篇文章主要为大家详细介绍了iOS实现波浪效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05
  • IOS视图控制器的生命周期实例详解

    IOS视图控制器的生命周期实例详解

    这篇文章主要介绍了IOS视图控制器的生命周期实例详解的相关资料,需要的朋友可以参考下
    2017-04-04
  • iOS SwiftUI 颜色渐变填充效果的实现

    iOS SwiftUI 颜色渐变填充效果的实现

    这篇文章主要介绍了iOS SwiftUI 颜色渐变填充效果的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-02-02
  • iOS如何自定义启动界面实例详解

    iOS如何自定义启动界面实例详解

    当我们打开一款应用程序的时候,首先映入眼帘的往往并不是程序的主界面,而是经过精心设计的欢迎界面,这个界面通常会停留几秒钟,然后消失。下面这篇文章主要给大家介绍了关于iOS如何自定义启动界面的相关资料,需要的朋友可以参考下。
    2017-12-12
  • iOS移动端软键盘弹起空白和滚动穿透问题解决方案

    iOS移动端软键盘弹起空白和滚动穿透问题解决方案

    这篇文章主要为大家介绍了iOS移动端软键盘弹起空白和滚动穿透问题解决方案,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-07-07
  • iOS开发中关键字const/static/extern、UIKIT_EXTERN的区别和用法

    iOS开发中关键字const/static/extern、UIKIT_EXTERN的区别和用法

    这篇文章主要介绍了iOS 关键字const/static/extern、UIKIT_EXTERN区别和用法,需要的朋友可以参考下
    2017-12-12
  • 简单说说iOS之WKWebView的用法小结

    简单说说iOS之WKWebView的用法小结

    iOS8.0之后我们使用 WebKit框架中的WKWebView来加载网页。这篇文章主要介绍了简单说说iOS之WKWebView的用法小结,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-01-01
  • iOS组件依赖避免冲突的小技巧分享

    iOS组件依赖避免冲突的小技巧分享

    这篇文章主要给大家介绍了关于iOS组件依赖避免冲突的小技巧,文中通过示例代码介绍的非常详细,对各位iOS开发者们具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-09-09

最新评论