iOS中的斷點續傳,對于做視頻類或下載的來說,比較重要,因為對于大文件,我們不可能迅速下載完成,這時候就需要我們本地將已經下載的給保存下來。
在此之前,網上流傳的大多是是NSURLConnection或者NSURLSessionDownloadTask這兩種方法來進行我們的需求,前者好像在iOS8左右就已經被蘋果拋棄了( 反正iOS9已被廢棄),后者用到的方法在下載的過程中,它并不會保存到本地,只會等到下載完成后,才會在本地顯示(當然了,你也可以用文件讀寫的方式將他的緩存文件自動寫入到本地),網上關于這兩種方法的demo我都研究了,簡單歸簡單,但是效果要么不好,要么就是被廢棄了,不能將就,得另尋辦法~
之后在網上輾轉,發現了一個大俠自己封裝的一個斷點續傳HSDownloadManager,他在內部用了NSURLSessionDataTask以及NSOutputStream流的方式,實現了可以多文件下載,并能同步緩存到本地,大家可以去github上下載: https://github.com/HHuiHao/HSDownloadManager
當然了,大蝦寫的也有bug,看了一下午他的內部實現原理,發現他的demo里,第一行刪除時候,任務隊列雖然刪除了,但是并沒有停止任務,導致了session的代理方法依然走,導致點擊清空時候程序會crash (我看網上也有些朋友提到過這個第三方,可都沒發現~),所以需要改一下。
具體內部實現在這不提了,有興趣的朋友下載這個第三方后(把那個bug改掉),可以自行查看,內部實現不難,一個單例,幾個方法。在這里提一下怎么簡單使用(快速集成到項目里),我自己寫了個簡單的demo,具體如下
1.將HSDownloadManager導入到項目里
2.定義基本控件屬性
@property (nonatomic, strong) UILabel *titleLabel;
@property (nonatomic, strong) UIButton *beginBtn;
@property (nonatomic, strong) UIButton *cancelBtn;
3.viewdidload中的簡單控件代碼
NSString * const downloadUrl = @"http://120.25.226.186:32812/resources/videos/minion_01.mp4";
- (void)viewDidLoad {
[super viewDidLoad];
NSLog(@"沙盒路徑: %@",NSHomeDirectory());
self.view.backgroundColor = [UIColor whiteColor];
self.title = @"斷點續傳";
_beginBtn = [UIButton buttonWithType:UIButtonTypeCustom];
_beginBtn.frame = CGRectMake(100, 100, 80, 30);
[_beginBtn setTitle:[self getTitleWithDownloadState:DownloadStateSuspended] forState:UIControlStateNormal];
_beginBtn.backgroundColor = [UIColor cyanColor];
[_beginBtn addTarget:self action:@selector(handleBtn:) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:_beginBtn];
_cancelBtn = [UIButton buttonWithType:UIButtonTypeCustom];
[_cancelBtn setTitle:@"刪除" forState:UIControlStateNormal];
_cancelBtn.frame = CGRectMake(100, 300, 80, 30);
_cancelBtn.backgroundColor = [UIColor cyanColor];
[_cancelBtn addTarget:self action:@selector(cancelBtn:) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:_cancelBtn];
_titleLabel = [[UILabel alloc] initWithFrame:CGRectMake(200, 150, 260, 20)];
[self.view addSubview:_titleLabel];
_titleLabel.text = @"當前下載進度:";
_titleLabel.textAlignment = 0;
[self refreshDataWithState:DownloadStateSuspended];
}
4.定義兩個方法,分別是刷新數據以及按鈕狀態的
#pragma mark 刷新數據(用來判斷程序剛進來時候下載的文件本地是否存在過,進度咋樣)
- (void)refreshDataWithState:(DownloadState)state
{
_titleLabel.text = [NSString stringWithFormat:@"當前下載進度:%.f%%", [[HSDownloadManager sharedInstance] progress:downloadUrl] * 100];
[self.beginBtn setTitle:[self getTitleWithDownloadState:state] forState:UIControlStateNormal];
NSLog(@"-----%f", [[HSDownloadManager sharedInstance] progress:downloadUrl]);
}
#pragma mark 按鈕狀態
- (NSString *)getTitleWithDownloadState:(DownloadState)state
{
switch (state) {
case DownloadStateStart:
return @"暫停";
case DownloadStateSuspended:
case DownloadStateFailed:
return @"開始";
case DownloadStateCompleted:
return @"完成";
default:
break;
}
}
5.最后點擊方法以及刪除方法
- (void)handleBtn:(UIButton *)btn {
[[HSDownloadManager sharedInstance] download:downloadUrl progress:^(NSInteger receivedSize, NSInteger expectedSize, CGFloat progress) {
dispatch_async(dispatch_get_main_queue(), ^{
_titleLabel.text = [NSString stringWithFormat:@"當前下載進度:%.2f%%", progress* 100];
});
} state:^(DownloadState state) {
NSLog(@"dsfsdgsd");
NSLog(@"state == %u", state);
dispatch_async(dispatch_get_main_queue(), ^{
[btn setTitle:[self getTitleWithDownloadState:state] forState:UIControlStateNormal];
});
}];
}
- (void)cancelBtn:(UIButton *)btn {
[[HSDownloadManager sharedInstance] deleteFile:downloadUrl];
_titleLabel.text = [NSString stringWithFormat:@"當前下載進度:%.f%%", [[HSDownloadManager sharedInstance] progress:downloadUrl] * 100];
}
效果如下:
怎么樣,是不是挺好使的~
如果你喜歡的話,就點個贊吧