iOS-Http断点续传
下载LOFTER客户端
IOS Http断点续传浅析
http实现断点续传的关键地方就是在httprequest中加入“Range”头。
//设置Range头,值:bytes=x-y;x:开始字节,y:结束字节,不指定则为文件末尾
[request addValue:@"bytes=500-" forHTTPHeaderField:@"Range"];
如果服务器正确响应的话,就可以顺利续传;如果服务器不支持,那就只能用其它方法了。
经过测试,服务器的不支持分为两种情况:
1.完全没响应
如果不处理会导致文件无法下载。
测试地址:http://dl_dir.qq.com/qqfile/qq/QQforMac/QQ_V2.4.2.dmg
发送请求后,过一段时间直接进入了didFailWithError的delegate;错误信息为time out。
针对这种情况可以做出的处理是:增加一个是否支持断点续传的标志。
具体:
第一次请求,开始字节为0,不用发送Range头,可以正常下载;
当下载中断,开始第二次请求,开始字节不为0,发送range头;
如果进入didFailWithError的delegate,就标明此链接不可以断点续传,每次请求前都清除缓存,保证开始的字节为0,不发送Range头。
2.无论发送Range的值是多少,服务器都会重新下载。
如果不处理,会导致续传过的文件出错。
测试地址:https://github.com/CocoaPods/CocoaPods/archive/master.zip
这种情况的处理方案是:
第一次收到响应的时候,就把文件的总大小记录下来;
以后每次收到响应的时候都比较一下下载长度和总大小是不是一样;
如果一样而且又存在缓存;就表明属于这种情况了;直接删掉缓存,重新下载。
下面是用NSURLConnection实现http断点续传的实例:
针对上面两种做了简单的处理,回调函数还有待添加
MXDownload.h文件:
#import <Foundation/Foundation.h>
@interface MXDownload : NSObject
//文件名路径
@property (nonatomic, readonly) NSString *filePath;
//是否正在下载的标志
@property (nonatomic, readonly) BOOL downloading;
//初始化
- (id)initWithUrlString:(NSString *)urlString;
//两个状态
- (void)start;
- (void)stop;
//清除缓存
- (void)clearCache;
@end
MXDownload.m文件:
#import "MXDownload.h"
#import "NSString+MX.h"
#define FILE_INFO_PLIST [NSString pathWithName:@"MXDownload/fileInfo.plist" directory:NSCachesDirectory]
@interface MXDownload (){
NSURLConnection *_urlConnection;
NSString *_urlString;
BOOL _downloading,_didAddRange,_shouldResume;
NSString *_fileName,*_filePath, *_tempFilePath;
NSFileHandle *_fileHandle;
unsigned long long _fileOffset,_fileSize;
}
@end
@implementation MXDownload
@synthesize downloading = _downloading;
@synthesize filePath = _filePath;
//初始化,顺便设置下载文件和下载临时文件路径
- (id)initWithUrlString:(NSString *)urlString{
self = [super init];
if (self){
_urlString = urlString;
_shouldResume = YES;
if (_urlString) {
_fileName = [_urlString MD5];
_filePath = [NSString pathWithName:[NSString stringWithFormat:@"MXDownload/%@",_fileName] directory:NSCachesDirectory];
_tempFilePath = [NSString stringWithFormat:@"%@.temp",_filePath];
}
}
return self;
}
//开始下载
- (void)start{
//如果正在下载,中断
if (_downloading) return;
//没有url,也中断
if (!_urlString) return;
//临时文件句柄
_fileHandle = [NSFileHandle fileHandleForWritingAtPath:_tempFilePath];
//获取本次请求下载开始的位置,如果文件不存在,就是0
_fileOffset = _fileHandle ? [_fileHandle seekToEndOfFile] : 0;
//初始化请求
NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:[NSURL URLWithString:_urlString]];
//设置缓存策略,很重要,因为文件是自己储存的,和缓存无关,所以要忽略缓存
//要不然第二次请求会出错
[request setCachePolicy:NSURLRequestReloadIgnoringLocalCacheData];
//最关键地方,设置Range头,值:bytes=x-y;x:开始字节,y:结束字节,不指定则为文件末尾
_didAddRange = NO;
if (_fileOffset != 0 && _shouldResume) {
[request addValue:[NSString stringWithFormat:@"bytes=%llu-",_fileOffset] forHTTPHeaderField:@"Range"];
_didAddRange = YES;
}
_urlConnection = [[NSURLConnection alloc] initWithRequest:request delegate:self];
[_urlConnection start];
_downloading = YES;
}
//结束下载
- (void)stop{
[_urlConnection cancel];
_urlConnection = nil;
[_fileHandle closeFile];
_downloading = NO;
}
//清除文件
- (void)clearCache{
if (_downloading) [self stop];
[[NSFileManager defaultManager] removeItemAtPath:_filePath error:nil];
[[NSFileManager defaultManager] removeItemAtPath:_tempFilePath error:nil];
}
#pragma mark -
#pragma mark NSURLConnectionDelegate
//接收到响应
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response{
//本次请求回来的文件大小
long long fileLength = response.expectedContentLength;
if (fileLength == NSURLResponseUnknownLength) [self stop];
NSData *existFileData = [[NSData alloc] initWithContentsOfFile:_filePath];
//检查文件是否已下载完成
if (existFileData && existFileData.length == fileLength) {
NSLog(@"之前已经下载好了");
[self stop];
}
else{
//保存文件的总大小
if (!_didAddRange){
NSMutableDictionary *dic = [NSMutableDictionary new];
[dic addEntriesFromDictionary:[NSDictionary dictionaryWithContentsOfFile:FILE_INFO_PLIST]];
[dic setValue:[NSNumber numberWithLongLong:fileLength] forKey:_fileName];
[dic writeToFile:FILE_INFO_PLIST atomically:YES];
}
NSFileManager *fileManager = [NSFileManager defaultManager];
//先清除掉旧的文件
[fileManager removeItemAtPath:_filePath error:nil];
//如果此次请求回来的大小等于文件的总大小而且临时文件又存在,则删除临时文件
//解决每次请求都是重新开始的问题
NSDictionary *dic = [NSDictionary dictionaryWithContentsOfFile:FILE_INFO_PLIST];
BOOL isTotalLength = fileLength == [[dic valueForKey:_fileName] longLongValue];
if ([fileManager fileExistsAtPath:_tempFilePath] && isTotalLength){
[fileManager removeItemAtPath:_tempFilePath error:nil];
}
//重新创建文件
if (![fileManager fileExistsAtPath:_tempFilePath]){
[fileManager createFileAtPath:_tempFilePath contents:nil attributes:nil];
_fileHandle = [NSFileHandle fileHandleForWritingAtPath:_tempFilePath];
_fileOffset = 0;
}
_fileSize = fileLength + _fileOffset;
//用_fileOffset可以检查是重新下载还是继续下载
NSLog(@"%@",_fileOffset ? @"继续下载" : @"开始下载");
}
}
//不断接收到数据
-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)aData{
//写入文件
[_fileHandle writeData:aData];
_fileOffset = [_fileHandle offsetInFile];
NSLog(@"下载进度: %lld / %lld",_fileOffset,_fileSize);
}
-(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error{
[self stop];
//如果不支持续传,删掉临时文件再试一次
if (_shouldResume) {
_shouldResume = NO;
[[NSFileManager defaultManager] removeItemAtPath:_tempFilePath error:nil];
[self start];
}
}
//完成
-(void)connectionDidFinishLoading:(NSURLConnection *)connection{
[[NSFileManager defaultManager] moveItemAtPath:_tempFilePath toPath:_filePath error:nil];
NSLog(@"下载完成");
[self stop];
}
@end
调用:
- (IBAction)startDownLoad:(id)sender{
if (_downloader == nil){
// _downloader = [[MXDownload alloc] initWithUrlString:@"https://github.com/CocoaPods/CocoaPods/archive/master.zip"];
_downloader = [[MXDownload alloc] initWithUrlString:@"http://dl_dir.qq.com/qqfile/qq/QQforMac/QQ_V2.4.2.dmg"];
// _downloader = [[MXDownload alloc] initWithUrlString:@"http://192.168.50.19:8080/vcont/wb.mp3"];
}
if (_downloader.downloading) {
[_downloader stop];
}
else{
[_downloader start];
}
}
iOS-Http断点续传的更多相关文章
- 总结iOS开发中的断点续传那些事儿
前言 断点续传概述 断点续传就是从文件赏赐中断的地方重新开始下载或者上传数据,而不是从头文件开始.当下载大文件的时候,如果没有实现断点续传功能,那么每次出现异常或者用户主动的暂停,都会从头下载,这样很 ...
- iOS开发之网络编程--4、NSURLSessionDataTask实现文件下载(离线断点续传下载) <进度值显示优化>
前言:根据前篇<iOS开发之网络编程--2.NSURLSessionDownloadTask文件下载>或者<iOS开发之网络编程--3.NSURLSessionDataTask实现文 ...
- iOS开发之网络编程--3、NSURLSessionDataTask实现文件下载(离线断点续传下载)
前言:使用NSURLSessionDownloadTask满足不这个需要离线断点续传的下载需求,所以这里就需要使用NSURLSessionDataTask的代理方法来处理下载大文件,并且实现离线断点续 ...
- iOS开发之网络编程--使用NSURLConnection实现大文件断点续传下载+使用输出流代替文件句柄
前言:本篇讲解,在前篇iOS开发之网络编程--使用NSURLConnection实现大文件断点续传下载的基础上,使用输出流代替文件句柄实现大文件断点续传. 在实际开发中,输入输出流用的比较少,但 ...
- iOS 应用开发中的断点续传实践总结
断点续传概述 断点续传就是从文件上次中断的地方开始重新下载或上传数据,而不是从文件开头.(本文的断点续传仅涉及下载,上传不在讨论之内)当下载大文件的时候,如果没有实现断点续传功能,那么每次出现异常或者 ...
- iOS中的下载管理器(支持断点续传)
在空闲时间自己编写了一个简单的iOS下载管理器.该管理器实现如下功能: 1.能够支持正常的下载,暂停,继续操作. 2.支持断点续传,实现暂停执行继续操作后,依然能正常将文件下载完成. 3.实现实时状态 ...
- ios 后台下载,断点续传总结
2018年12月05日 16:09:00 weixin_34101784 阅读数:5 https://blog.csdn.net/weixin_34101784/article/details/875 ...
- iOS 文件下载及断点续传
ios的下载我们可以使用的方法有:NSData.NSURLConnection.NSURLSession还有第三方框架AFNetworking和ASI 利用NSData方法和NSURLConnecti ...
- 数据存储之iOS断点续传
iOS里面实现断点续传 第三方框架之AFN 代码实现 一.iOS里面实现断点续传 1⃣️AFN基于NSURL 1.性能和稳定性略差.针对JSON.XML.Plist和Image四种数据结构封装了各自处 ...
随机推荐
- sem_open 信号量操作
sem_open,计算机用语.意思是创建并初始化有名信号量或打开一个已存在的有名信号量 sem_init:初始化信号量sem_t,初始化的时候可以指定信号量的初始值,以及是否可以在多进程间共享. se ...
- 将CHROME WEBSTORE里的APPS和扩展下载到本地 转载自:http://ro-oo.diandian.com/post/2011-05-28/1103036
Chrome Webstore 自动改版后就不能再直接下载到本地... 下载地址: https://clients2.google.com/service/update2/crx?response=r ...
- C指针乱记
//int a[3][4] = { { 66, 2, 3, 4 }, { 5, 6, 7, 8 }, { 9, 10, 11, 12 } }; //读取二维数组任意元素hint int(*)a[4] ...
- Mysql批量更新的一个坑-&allowMultiQueries=true允许批量更新(转)
实际上,我们经常会遇到这样的需求,那就是利用Mybatis批量更新或者批量插入,但是,实际上即使Mybatis完美支持你的sql,你也得看看你说操作的数据库是否支持,而阿福,最近就遇到这样的一个坑. ...
- maven+struts2例子
本文在开发第一个maven示例的基础上进行扩展. 第一个maven示例目录结构如下: 扩展后目录为: 打开pom.xml 在WEB-INF文件夹下新建web.xml: <?xml version ...
- [HNOI2015]开店 简要题解
主席树. 推下式子,发现点的深度和好算,lca深度和不好算. lca深度之和有个套路:先给a到根路径+1,再算b到根的和. 如果可以离线,即LNOI的LCA.本题强制在线,可持久化. 由于区间修改,使 ...
- @PathVariable,@RequestParam, @RequestBody
https://www.cnblogs.com/guoyinli/p/7056146.html https://www.cnblogs.com/zeroingToOne/p/8992746.html ...
- 为什么margin:0 auto不能用于inline-block元素
前言:今天一个实习生问我,为什么他对图片使用了margin:0 auto,但图片却没有居中,我让他换成对父元素使用text-align:center即可.为什么margin:0 auto对图片不起作用 ...
- HDU3076 ssworld VS DDD
嘟嘟嘟 友情提示:数据把\(hp1\)和\(hp2\)弄反了! 进入正题. 这题还是比较好想,令\(dp[i][j]\)表示第一个人赢了\(i\)场,第二个人赢了\(j\)的概率,转移就是分别考虑这一 ...
- Smali基础知识
Smali是用于Dalvik(Android虚拟机)的反汇编程序实现 汇编工具(将Smali代码汇编为dex文件)为smali.jar baksmali.jar则是反汇编程序 地址:https://b ...