前言:根据前篇《iOS开发之网络编程--2、NSURLSessionDownloadTask文件下载》或者《iOS开发之网络编程--3、NSURLSessionDataTask实现文件下载(离线断点续传下载)》,都遗留了一个细节未处理的问题,那就是在离线断点下载的过程中,当应用程序重新启动之后,进度条的进度值默认没有设置为之前已经下载的进度,根据基本公式"当前进度值 = 已经下载的数据长度 ÷ 最终下载完的数据总长度",已经下载的数据长度可以由沙盒中已经下载的那部分数据获取,但是最终下载完的数据总长度就需要通过网络返回的信息了,但是别忘了,每一次重新启动应用程序初始状态默认都是暂停下载,或者是断网的情况下无法请求网络数据,那么如何获取这个"最终下载完的数据总长度"呢?

本篇还涉及到在子线程创建下载任务,然后通过线程通知UI主线程更新进度条控件显示进度。因为delegateQueue这个属性可以设置主队列线程或者是子队列线程。

先看看效果:

问题解决:"最终下载完的数据总长度"可以在首次从0开始下载的时候通过网络获取,然后将其"最终下载完的数据总长度"这个值存储在缓存中的某个文件(这个文件可以是字典等等)中,等待下一次获取。

       而我则采用的方法是将这个"最终下载完的数据总长度"作为文件的属性添加进文件属性列表中,以备下一次读取的时候,获得到这个文件之后,就可以读取该文件的属性列表中的"最终下载完

     的数据总长度"的属性和属性值。

在这里不得不先介绍一个工具类,读者可以通过本人的另一篇博文随笔先了解其功能:iOS开发 -- 为本地文件添加自定义属性的工具类

为本地文件添加属性之后,可以打印看的到:

本人花了点时间将网络下载这部分简单的封装成了一个工具类:DownloadTool,下面就展示源码:

DownloadTool.h

 #import <Foundation/Foundation.h>

 // 定义一个block用来传递进度值
typedef void (^SetProgressValue)(float progressValue); @interface DownloadTool : NSObject /** 创建下载工具对象 */
+ (instancetype)DownloadWithURLString:(NSString*)urlString setProgressValue:(SetProgressValue)setProgressValue;
/** 开始下载 */
-(void)startDownload;
/** 暂停下载 */
-(void)suspendDownload; @end

DownloadTool.m

 #import "DownloadTool.h"

 #import "ExpendFileAttributes.h"

 #define Key_FileTotalSize @"Key_FileTotalSize"

 @interface DownloadTool () <NSURLSessionDataDelegate>
/** Session会话 */
@property (nonatomic,strong)NSURLSession *session;
/** Task任务 */
@property (nonatomic,strong)NSURLSessionDataTask *task;
/** 文件的全路径 */
@property (nonatomic,strong)NSString *fileFullPath;
/** 传递进度值的block */
@property (nonatomic,copy) SetProgressValue setProgressValue;
/** 当前已经下载的文件的长度 */
@property (nonatomic,assign)NSInteger currentFileSize;
/** 输出流 */
@property (nonatomic,strong)NSOutputStream *outputStream;
/** 不变的文件总长度 */
@property (nonatomic,assign)NSInteger fileTotalSize;
@end @implementation DownloadTool + (instancetype)DownloadWithURLString:(NSString*)urlString setProgressValue:(SetProgressValue)setProgressValue{
DownloadTool* download = [[DownloadTool alloc] init];
download.setProgressValue = setProgressValue;
[download getFileSizeWithURLString:urlString];
[download creatDownloadSessionTaskWithURLString:urlString];
NSLog(@"%@",download.fileFullPath);
return download;
}
// 刚创建该网络下载工具类的时候,就需要查询本地是否有已经下载的文件,并返回该文件已经下载的长度
-(void)getFileSizeWithURLString:(NSString*)urlString{
// 创建文件管理者
NSFileManager* fileManager = [NSFileManager defaultManager];
// 获取文件各个部分
NSArray* fileComponents = [fileManager componentsToDisplayForPath:urlString];
// 获取下载之后的文件名
NSString* fileName = [fileComponents lastObject];
// 根据文件名拼接沙盒全路径
NSString* fileFullPath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:fileName];
self.fileFullPath = fileFullPath; NSDictionary* attributes = [fileManager attributesOfItemAtPath:fileFullPath
error:nil];
// 如果有该文件,且为下载没完成,就直接拿出该文件的长度设置进度值,并设置当前的文件长度
NSInteger fileCurrentSize = [attributes[@"NSFileSize"] integerValue];
// 如果文件长度为0,就不需要计算进度值了
if (fileCurrentSize != ) {
// 获取最终的文件中长度
NSInteger fileTotalSize = [[ExpendFileAttributes stringValueWithPath:self.fileFullPath key:Key_FileTotalSize] integerValue];
self.currentFileSize = fileCurrentSize;
self.fileTotalSize = fileTotalSize;
// 设置进度条的值
self.setProgressValue(1.0 * fileCurrentSize / fileTotalSize);
}
NSLog(@"当前文件长度:%lf" , self.currentFileSize * 1.0);
}
#pragma mark - 创建网络请求会话和任务,并启动任务
-(void)creatDownloadSessionTaskWithURLString:(NSString*)urlString{
//判断文件是否已经下载完毕
if (self.currentFileSize == self.fileTotalSize && self.currentFileSize != ) {
NSLog(@"文件已经下载完毕");
return;
}
NSURLSession* session =
[NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
delegate:self
delegateQueue:[[NSOperationQueue alloc]init]];
NSURL* url = [NSURL URLWithString:urlString];
NSMutableURLRequest* request = [NSMutableURLRequest requestWithURL:url];
//2.3 设置请求头
NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.currentFileSize];
[request setValue:range forHTTPHeaderField:@"Range"];
NSURLSessionDataTask* task = [session dataTaskWithRequest:request];
self.session = session;
self.task = task;
} #pragma mark - 控制下载的状态
// 开始下载
-(void)startDownload{
[self.task resume];
}
// 暂停下载
-(void)suspendDownload{
[self.task suspend];
}
#pragma mark - NSURLSessionDataDelegate 的代理方法
// 收到响应调用的代理方法
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:
(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler{
NSLog(@"执行了收到响应调用的代理方法");
// 创建输出流,并打开流
NSOutputStream* outputStream = [[NSOutputStream alloc] initToFileAtPath:self.fileFullPath append:YES];
[outputStream open];
self.outputStream = outputStream;
// 如果当前已经下载的文件长度等于0,那么就需要将总长度信息写入文件中
if (self.currentFileSize == ) {
NSInteger totalSize = response.expectedContentLength;
NSString* totalSizeString = [NSString stringWithFormat:@"%ld",totalSize];
[ExpendFileAttributes extendedStringValueWithPath:self.fileFullPath key:Key_FileTotalSize value:totalSizeString];
// 别忘了设置总长度
self.fileTotalSize = totalSize;
}
// 允许收到响应
completionHandler(NSURLSessionResponseAllow);
}
// 收到数据调用的代理方法
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data{
NSLog(@"执行了收到数据调用的代理方法");
// 通过输出流写入数据
[self.outputStream write:data.bytes maxLength:data.length];
// 将写入的数据的长度计算加进当前的已经下载的数据长度
self.currentFileSize += data.length;
// 设置进度值
NSLog(@"当前文件长度:%lf,总长度:%lf",self.currentFileSize * 1.0,self.fileTotalSize * 1.0);
NSLog(@"进度值: %lf",self.currentFileSize * 1.0 / self.fileTotalSize);
// 获取主线程
NSOperationQueue* mainQueue = [NSOperationQueue mainQueue];
[mainQueue addOperationWithBlock:^{
self.setProgressValue(self.currentFileSize * 1.0 / self.fileTotalSize);
}];
}
// 数据下载完成调用的方法
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error{
// 关闭输出流 并关闭强指针
[self.outputStream close];
self.outputStream = nil;
// 关闭会话
[self.session invalidateAndCancel];
NSLog(@"%@",[NSThread currentThread]);
}
-(void)dealloc{
}
@end

使用示例源码:

 #import "ViewController.h"
#import "RainbowProgress.h" #import "DownloadTool.h" #define MP4_URL_String @"http://120.25.226.186:32812/resources/videos/minion_12.mp4" @interface ViewController ()
@property (weak, nonatomic) IBOutlet UILabel *showDownloadState;
/** 彩虹进度条 */
@property (nonatomic,weak)RainbowProgress *rainbowProgress;
/** 网络下载工具对象 */
@property (nonatomic,strong)DownloadTool *download;
@end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
[self setSelfView];
[self addProgress];
[self addDownload]; }
// 启动和关闭的网络下载开关
- (IBAction)SwitchBtn:(UISwitch *)sender {
if (sender.isOn) {
self.showDownloadState.text = @"开始下载";
[self.download startDownload];
}else{
self.showDownloadState.text = @"暂停下载";
[self.download suspendDownload];
}
}
#pragma mark - 设置控制器View
-(void)setSelfView{
self.view.backgroundColor = [UIColor blackColor];
}
#pragma mark - 添加彩虹进度条
-(void)addProgress{
// 创建彩虹进度条,并启动动画
RainbowProgress* rainbowProgress = [[RainbowProgress alloc] init];
[rainbowProgress startAnimating];
[self.view addSubview:rainbowProgress];
self.rainbowProgress = rainbowProgress;
}
#pragma mark - 创建网络下载任务
-(void)addDownload{
DownloadTool* download = [DownloadTool DownloadWithURLString:MP4_URL_String setProgressValue:^(float progressValue) {
self.rainbowProgress.progressValue = progressValue;
}];
self.download = download;
} #pragma mark - 设置状态栏样式
-(UIStatusBarStyle)preferredStatusBarStyle{
return UIStatusBarStyleLightContent;
} @end

效果图(中间有个过程是重新启动应用程序看看进度条显示的效果,然后继续测试开始下载和暂停下载):

百度云分享源码链接: http://pan.baidu.com/s/1eRwRkZo 密码: 787n

转载请注明出处:http://www.cnblogs.com/goodboy-heyang/p/5200873.html ,尊重劳动成果。

iOS开发之网络编程--4、NSURLSessionDataTask实现文件下载(离线断点续传下载) <进度值显示优化>的更多相关文章

  1. iOS开发之网络编程--使用NSURLConnection实现大文件断点续传下载+使用输出流代替文件句柄

    前言:本篇讲解,在前篇iOS开发之网络编程--使用NSURLConnection实现大文件断点续传下载的基础上,使用输出流代替文件句柄实现大文件断点续传.    在实际开发中,输入输出流用的比较少,但 ...

  2. iOS开发之网络编程--3、NSURLSessionDataTask实现文件下载(离线断点续传下载)

    前言:使用NSURLSessionDownloadTask满足不这个需要离线断点续传的下载需求,所以这里就需要使用NSURLSessionDataTask的代理方法来处理下载大文件,并且实现离线断点续 ...

  3. iOS开发之网络编程--2、NSURLSessionDownloadTask文件下载

    本文内容大纲: 1.回顾NSURLSessionTask 2.NSURLSessionDownloadTask大文件之block下载 3.NSURLSessionDownloadTask大文件之代理方 ...

  4. iOS开发之网络编程--使用NSURLConnection实现大文件下载

    主要思路(实现下载数据分段写入缓存中) 1.使用NSURLConnectionDataDelegate以及代理方法.2.在成功获取响应的代理方法中,获得沙盒全路径,并在该路径下创建空文件和文件句柄.3 ...

  5. iOS开发之网络编程--5、NSURLSessionUploadTask+NSURLSessionDataDelegate代理上传

    前言:关于NSURLSession的主要内容快到尾声了,这里就讲讲文件上传.关于文件上传当然就要使用NSURLSessionUploadTask,这里直接讲解常用的会和代理NSURLSessionDa ...

  6. iOS开发之网络编程--使用NSURLConnection实现文件上传

    前言:使用NSURLConnection实现文件上传有点繁琐.    本文并没有介绍使用第三方框架上传文件. 正文: 这里先提供用于编码测试的接口:http://120.25.226.186:3281 ...

  7. iOS开发之网络编程--小文件下载

    文件下载方式: 如果下载的文件比较小,下载方式: 直接用NSData的 +(id)dataWithContentsOfURL:(NSURL*)url; 利用NSURLConnection发送一个HTT ...

  8. IOS开发之网络编程--文件压缩和解压缩

    前言: QQ表情包就用到了解压缩,从网络下载的那么多表情文件格式并不是一个一个图片文件,而是多个图片压缩而成的表情压缩包.下面介绍的是iOS开发中会用到的压缩和解压缩的第三方框架的使用. 注意: 这个 ...

  9. iOS开发之网络编程--XCode7 更新以来需要手动设置的内容

    XCode7 更新以来,默认是不允许加载一些http网络请求,是因为现在网络大部分使用更安全的https协议头. 所以,iOS网络编程如果出现请求无效,事先考虑是否设置一下了以下操作:

随机推荐

  1. Tips6:用[HideInInspector]在Inspector中隐藏变量

    你曾想要把某些公有变量在Inspetor面板中隐藏起来吗,这样很简单就能做到. 让我们来看以下代码: using UnityEngine; public class HidingScript : Mo ...

  2. Mount NAS Storage in Linux Overview 转载

    Mount NAS Storage in Linux Overview Mounting your NAS Storage to a device that runs on a Linux-based ...

  3. [数据库]sql之行顺序

    这个文章主要是防止我忘了sql的执行顺序,解释的东西我都没怎么看懂.数据库渣如我- 逻辑查询处理阶段简介 FROM:对FROM子句中的前两个表执行笛卡尔积(Cartesian product)(交叉联 ...

  4. stl的优先级队列

    #include <iostream> #include <vector> #include <queue> using namespace std; class ...

  5. mysql一个事务中有DDL语句的binlog情况

      在autocommit=1的情况下,开启一个事务,如果里面有DDL语句,那么事务开始到DDL语句之间的DML语句都会被提交.再开启新的事务.可以从binlog中看出   session语句: 09 ...

  6. servlet中的细节

    Get方法有大小限制:1024个字符.这些信息使用 Query_String头传递,并通过Query_String环境变量访问.Post方法:请求体信息使用FromData头传递.读取所有表单参数:g ...

  7. ScriptManager和UpdatePanel控件实现局部刷新

    ScriptManager和UpdatePanel控件联合使用可以实现页面异步局部更新的效果.其中的UpdatePanel就是设置页面中异步局部更新区域,它必须依赖于ScriptManager存在,因 ...

  8. LeetCode128:Longest Consecutive Sequence

    题目: Given an unsorted array of integers, find the length of the longest consecutive elements sequenc ...

  9. 邻接矩阵c源码(构造邻接矩阵,深度优先遍历,广度优先遍历,最小生成树prim,kruskal算法)

    matrix.c #include <stdio.h> #include <stdlib.h> #include <stdbool.h> #include < ...

  10. Python正则表达式模块(re模块)

    Python是我接触到的第一门编程语言,虽然它足够简单,但是对于当时刚刚接触编程语言的我来说还是有些难度的,于是只是了解了一些Python的基本语法,稍微深入一点的地方都没怎么了解.不过,到现在为止, ...