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];
    }
}

时间: 2024-08-26 15:36:57

IOS Http断点续传浅析的相关文章

ios 实现断点续传 一 nsurlconnection

NSUrlConnection实现断点续传的关键是自定义http request的头部的range域属性. Range头域 Range头域可以请求实体的一个或者多个子范围.例如, 表示头500个字节:bytes=0-499 表示第二个500字节:bytes=500-999 表示最后500个字节:bytes=-500 表示500字节以后的范围:bytes=500- 第一个和最后一个字节:bytes=0-0,-1 同时指定几个范围:bytes=500-600,601-999 但是服务器可以忽略此请求

IOS中CoreData浅析

CoreData简介: 什么是CoreData? Core Data是iOS5之后才出现的一个框架,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象.在此数据操作期间,我们不需要编写任何SQL语句 CoreData 不能执行SQL语句 取而代之,操作的是对象.而常用的三方库 FMDB SQLite 可以直接SQL语句 CoreData和数据库有什么区别? CoreData是一个苹果原生的框架,它拥有像数

iOS钥匙串Keychain浅析

根据苹果的介绍,iOS设备中的Keychain是一个安全的存储容器,可以用来为不同应用保存敏感信息比如用户名,密码,网络密码,认证令牌.苹果自己用keychain来保存WiFi密码,VPN凭证等.它是一个SQLite数据库,位于/private/var/Keychains/keychain-2.db,其保存的所有数据都是加密过的. 比如在Mac的LaunchPad中,就有一个钥匙串访问,打开就可以看到存储有很多重要的信息: . 开发者通常会希望能够利用操作系统提供的功能来保存凭证(credent

IOS后台运行浅析

在之前的文章<App States and Multitasking IOS应用程序状态和多任务>说到IOS程序有前台后台之分.在IOS程序进入后台之后,程序就会不执行代码,如果非要有什么必须执行的过程,可以申请大约600s的时间,如果在这段时间内程序还没有完成则会被系统杀死.出现:has active assertions beyond permitted time. 更多IOS后台内容参考:<IOS7 Background Fetch后台应用程序刷新> <IOS 7四种后

第二回《iOS之Block浅析》

====================================================================== |   转载须注明博客地址:http://www.cnblogs.com/devappnow            | |   请尊重原创.尊重别人的劳动成果.如果读者你觉得有帮助.也可以通过任意的方式给博主鼓励(推荐,评论,邮件等等).    | ======================================================

iOS 多语言 浅析

什么是本地化处理? 本地化处理就是我们的应用程序有可能发布到世界的很多国家去,因为每个国家应用的语言是不一样的,所以我们要把我们的应用程序的语言要进行本地化处理一下. 本地化处理需要处理那些文件? (1):本地化应用程序名称(我的的项目名称) (2):本地化字符串处理(项目中所涉及的字符串) (3):本地化图片 (4):XIB文件本地化处理 (5):其他文件 下面进行本地化处理: 1 . 设置我们的工程支持多语言 为什么:因为只有设置成支持多语言了,我们才能在展示项目名称和项目中的字符串等将语言

iOS内存警告浅析

在开发中,内存的资源是很宝贵的.所以合理的管理项目的内存是恨重要的. 在项目中,最消耗内存的,莫过于图片.我们都知道,最常用的获取图片的方式是[UIImage imageName:@"image"];是这种方式,这种方式有一个机制,就是自动的有缓存.还有一种常用的方式是:[UIImage imageWithContentsOfFile:"fileName"];这种方式是没有缓存的.总结:由上可知,如果你想获娶大的图片,再加上不常用到此图片,建议使用第二种方式来加载图

ios开发--KVO浅析

目标:监听NSMutableArray对象中增加了什么 代码如下: C代码   - (void)viewDidLoad { [super viewDidLoad]; self.dataArray = [NSMutableArray arrayWithObject:@"1"]; [self addObserver:self forKeyPath:@"dataArray" options:NSKeyValueObservingOptionNew | NSKeyValue

IOS ReactiveCocoa

一 前提: 在iOS开发过程中,当某些事件响应时,需处理的某些业务逻辑 Eg. 按钮点击:action ScrollView滚动:delegate 属性值改变:KVO ReactiveCocoa为事件提供了很多的处理方法,而且利用RAC处理事件很方便,可以把要处理的事件和监听代码放在一起,这样便于管理,不需要跳到对应的方法中,非常符合高聚合,低耦合的思想 二 ReactiveCocoa整体结构介绍: 1 信号源(事件流)RACStream 2 订阅者 RACSubscriber:在使用-subs