【待整理】IOS开发之下载

一、传统的下载文件的方式  

- (void)downloaderWithUrl:(NSURL *)url
{
    NSURLRequest *request = [NSURLRequest requestWithURL:url];

    [NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {

        if (!connectionError)
        {
            NSHTTPURLResponse *urlResponse = (NSHTTPURLResponse *)response;
            if (urlResponse.statusCode == 200 || urlResponse.statusCode == 304)
            {
                [data writeToFile:@"/Users/gxiangzi/Desktop/a.mp4" atomically:YES];
                NSLog(@"OK");
            }
        }else
        {
            NSLog(@"错误");
        }
    }];
}

但是这种方式存在着弊端:

  1> 用不无法跟踪下载的进度,无法直观的显示下载进度

  2> 文件下载的时候,都会先下载到内存之中,导致内存 > 所需文件的大小长度,会造成程序崩溃等现象(可以查看内存的耗费)

二、代理的方式下载程序(NSURLConnectionDownloadDelegate)

#import "GXDownloader.h"

@interface GXDownloader () <NSURLConnectionDownloadDelegate>

@end

@implementation GXDownloader

- (void)downloaderWithUrl:(NSURL*)url
{
    NSURLRequest* request = [NSURLRequest requestWithURL:url];
    [[NSURLConnection alloc] initWithRequest:request delegate:self];
}

#pragma mark - NSURLConnectionDownloadDelegate

/**
 bytesWritten: 表示下载的数据字节数
 totalBytesWritten:总的已经下载的字节数
 expectedTotalBytes:期望的总的字节数
 */
- (void)connection:(NSURLConnection*)connection didWriteData:(long long)bytesWritten totalBytesWritten:(long long)totalBytesWritten expectedTotalBytes:(long long)expectedTotalBytes
{
    float progress = (float)totalBytesWritten / expectedTotalBytes;
    NSLog(@"%f", progress);
}

/**
 *  @param connection
 *  @param totalBytesWritten : 总的已经下载的字节数
 *  @param expectedTotalBytes : 期望总的字节数
 */
- (void)connectionDidResumeDownloading:(NSURLConnection*)connection totalBytesWritten:(long long)totalBytesWritten expectedTotalBytes:(long long)expectedTotalBytes
{
}

/**
 *  @param connection
 *  @param destinationURL:下载完的目标路径
 */
- (void)connectionDidFinishDownloading:(NSURLConnection*)connection destinationURL:(NSURL*)destinationURL
{
    NSLog(@"%@", destinationURL);
}

优点:

  1> 该种方式可以跟踪文件下载的进度;

  2> 也可以解决内存消耗过大的问题

弊端:  这丫的 下载的文件没有 ,就是虽然你显示下载完成,但是你本地找不到该文件------没有任何卵用

三、第三种方式--代理(NSURLConnectionDataDelegate)

//
//  GXDownloader.m
//  02-下载
//
//  Created by gxiangzi on 15/8/21.
//  Copyright (c) 2015年 hqu. All rights reserved.
//

#import "GXDownloader.h"

@interface GXDownloader () <NSURLConnectionDataDelegate>

// 文件的总长度
@property (nonatomic, assign) long long expectedContentLength;

// 已经接受到文件的长度
@property (nonatomic, assign) long long receivedContentLength;

// 已经接收到的文件的总长度
@property (nonatomic, strong) NSMutableData* receivedData;

@end

@implementation GXDownloader

- (void)downloaderWithUrl:(NSURL*)url
{
    NSURLRequest* request = [NSURLRequest requestWithURL:url];
    [[NSURLConnection alloc] initWithRequest:request delegate:self];
}

#pragma mark - NSURLConnectionDataDelegate

/**
 *  已经接受到响应头
 */
- (void)connection:(NSURLConnection*)connection didReceiveResponse:(NSURLResponse*)response
{
    self.expectedContentLength = response.expectedContentLength;
}

/**
 *  文件接受到一点的时候就是用
 */
- (void)connection:(NSURLConnection*)connection didReceiveData:(NSData*)data
{
    // 把二进制数据进行拼接
    [self.receivedData appendData:data];
    // 拼接长度
    self.receivedContentLength += data.length;

    float progress = (float)self.receivedContentLength / self.expectedContentLength;

    NSLog(@"下载进度: %.2f",progress);

}

/**
 *  下载出错的时候调用
 */
- (void)connection:(NSURLConnection*)connection didFailWithError:(NSError*)error
{
}

/**
 *  当下载完成的时候调用
 */
- (void)connectionDidFinishLoading:(NSURLConnection*)connection
{
    NSLog(@"下载完成");

    // IOS 的文件路径,只能讲软件操作的数据放在沙盒内
    NSString *path = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
    NSString *fileName = [path stringByAppendingPathComponent:@"love.mp4"];

    NSLog(@"%@",fileName);

    [self.receivedData writeToFile:fileName atomically:YES];

}

#pragma mark -懒加载
- (NSMutableData*)receivedData
{
    if (!_receivedData) {
        _receivedData = [NSMutableData data];
    }
    return _receivedData;
}

@end

优点:

  1> 可以检测到进度的问题

  2> 文件的写入的路径之中,文件可以存在

弊端:

  占用的内存太大  

为了解决上面的问题: 引入了NSHandle  和 NSOutputStream ------下一点 写入文件一点

四、NSHandle的使用

//
//  GXDownloader.m
//  02-下载
//
//  Created by gxiangzi on 15/8/21.
//  Copyright (c) 2015年 hqu. All rights reserved.
//

#import "GXDownloader.h"

@interface GXDownloader () <NSURLConnectionDataDelegate>

// 文件的总长度
@property (nonatomic, assign) long long expectedContentLength;

// 已经接受到文件的长度
@property (nonatomic, assign) long long receivedContentLength;

@end

@implementation GXDownloader

- (void)downloaderWithUrl:(NSURL*)url
{
    NSURLRequest* request = [NSURLRequest requestWithURL:url];
    [[NSURLConnection alloc] initWithRequest:request delegate:self];
}

#pragma mark - NSURLConnectionDataDelegate

/**
 *  已经接受到响应头
 */
- (void)connection:(NSURLConnection*)connection didReceiveResponse:(NSURLResponse*)response
{
    self.expectedContentLength = response.expectedContentLength;
}

/**
 *  文件接受到一点的时候就是用
 */
- (void)connection:(NSURLConnection*)connection didReceiveData:(NSData*)data
{
    // 拼接长度
    self.receivedContentLength += data.length;

    float progress = (float)self.receivedContentLength / self.expectedContentLength;
    NSLog(@"%f",progress);

    NSString* path = @"/Users/gxiangzi/Desktop/a.mp4";

    NSFileHandle* handle = [NSFileHandle fileHandleForWritingAtPath:path];
    // 如果handle  不会帮我们创建文件
    if (handle == nil) {
        [data writeToFile:path atomically:YES];
    }
    else {
        //让offset指向文件的末尾,在末尾处追加数据
        [handle seekToEndOfFile];
        //写输入
        [handle writeData:data];
        //关闭  (等文件下载完成关闭更好)
        [handle closeFile];
    }

}

/**
 *  当下载完成的时候调用
 */
- (void)connectionDidFinishLoading:(NSURLConnection*)connection
{
    NSLog(@"下载完成");
}

@end

原理:下载一点数据 写一点数据

效果:能够达到占用很少内存

五、NSOutputStream的使用

//
//  GXDownloader.m
//  02-下载
//
//  Created by gxiangzi on 15/8/21.
//  Copyright (c) 2015年 hqu. All rights reserved.
//

#import "GXDownloader.h"

@interface GXDownloader () <NSURLConnectionDataDelegate>

// 文件的总长度
@property (nonatomic, assign) long long expectedContentLength;

// 已经接受到文件的长度
@property (nonatomic, assign) long long receivedContentLength;

@property (nonatomic, strong) NSOutputStream* stream;

@end

@implementation GXDownloader

- (void)downloaderWithUrl:(NSURL*)url
{
    NSURLRequest* request = [NSURLRequest requestWithURL:url];
    [[NSURLConnection alloc] initWithRequest:request delegate:self];
}

#pragma mark - NSURLConnectionDataDelegate

/**
 *  已经接受到响应头
 */
- (void)connection:(NSURLConnection*)connection didReceiveResponse:(NSURLResponse*)response
{
    self.expectedContentLength = response.expectedContentLength;

    NSString* path = @"/Users/gxiangzi/Desktop/a.mp4";

    // 注意 此处的url 为本地的路径
    // 前面加 file:// 无效
    // 只能使用 [NSURL fileURLWithPath:path]
   self.stream = [NSOutputStream outputStreamWithURL:[NSURL fileURLWithPath:path] append:YES];

    [self.stream open];

}

/**
 *  文件接受到一点的时候就是用
 */
- (void)connection:(NSURLConnection*)connection didReceiveData:(NSData*)data
{
    // 拼接长度
    self.receivedContentLength += data.length;

    float progress = (float)self.receivedContentLength / self.expectedContentLength;
    NSLog(@"%f", progress);

    [self.stream write:data.bytes maxLength:data.length];
}

/**
 *  当下载完成的时候调用
 */
- (void)connectionDidFinishLoading:(NSURLConnection*)connection
{
    NSLog(@"下载完成");
    [self.stream close];
}

- (NSOutputStream *)stream
{
    if (_stream == nil)
    {
        _stream = [[NSOutputStream alloc] init];
    }
    return _stream;
}

@end
时间: 2024-12-20 23:29:17

【待整理】IOS开发之下载的相关文章

整理iOS开发常用的第三方资源

一:第三方插件 1:基于响应式编程思想的oc 地址:https://github.com/ReactiveCocoa/ReactiveCocoa 2:hud提示框 地址:https://github.com/jdg/MBProgressHUD 3:XML/HTML解析 地址:https://github.com/topfunky/hpple 4:有文字输入时,能根据键盘是否弹出来调整自身显示内容的位置 地址:https://github.com/michaeltyson/TPKeyboardAv

李洪强iOS开发之下载

// // //  LHQDownLoader.m //  A21 - 李洪强 - 下载 // //  Created by vic fan on 16/7/3. //  Copyright © 2016年 李洪强. All rights reserved. // #import "LHQDownLoader.h" @interface LHQDownLoader ()<NSURLSessionDataDelegate> @end @implementation LHQDo

整理IOS开发常用小Tips

在NavigationController中 重写 show/pushViewController方法统一在导航栏中添加按钮 - (void)showViewController:(UIViewController *)vc sender:(id)sender { if (self.viewControllers.count > 0) { //首页不显示跳转可以通过只添加leftBtn来间接修改所有返回按钮的图片 vc.hidesBottomBarWhenPushed = YES; vc.nav

文顶顶iOS开发博客链接整理及部分项目源代码下载

文顶顶iOS开发博客链接整理及部分项目源代码下载 网上的iOS开发的教程很多,但是像cnblogs博主文顶顶的博客这样内容图文并茂,代码齐全,示例经典,原理也有阐述,覆盖面宽广,自成系统的系列教程却很难找.如果你是初学者,在学习了斯坦福iOS7公开课和跟着文顶顶的博客做项目之后,最快只需要2个月时间,就基本可以独立完成iOS App的开发工作.有经验的开发者也可以在该博客中寻找代码片段进行学习借鉴,必有所收获. 在此也向@文顶顶 表示严重感谢! 由于文顶顶博客博文繁多,每次找文章需要频繁的翻页,

【热门收藏】iOS开发人员必看的精品资料(100个)——下载目录

iPhone.iPad产品风靡全球,巨大的用户群刺激着iOS软件开发需求,然而国内人才缺口很大,正处于供不应求的状态,ios开发前景大好.我们整理了51CTO下载中心100份热门的ios开发资料,做了一个地址收藏目录,方便大家选择性下载.里边均是好资料,相信对学习ios开发有所帮助.目录下载地址:http://down.51cto.com/data/416672 附件预览如下: iOS游戏开发之Unity3D引擎系列 http://down.51cto.com/zt/399 iPadian软件:

iOS开发下载文件速度计算

当我们写下载界面的时候,需要向用户展示每秒下载多少KB,这个时候就需要计算速度.如下: 我用的是AFNetworking来做下载的,我们拿AFHTTPRequestOperation来举列,AFHTTPRequestOperation中有个方法: - (void)setDownloadProgressBlock:(void (^)(NSUInteger bytesRead, long long totalBytesRead, long long totalBytesExpectedToRead)

iOS开发UI篇—推荐两个好用的Xcode插件(提供下载链接)

iOS开发UI篇—推荐两个好用的Xcode插件(提供下载链接) 这里推荐两款好用的Xcode插件,并提供下载链接. 一.插件和使用如下: 1.两款插件 对项目中图片提供自动提示功能的插件:KSImageNamed-Xcode-master 提供快速创建自动注释:VVDocumenter-Xcode-master 2.使用介绍: (1)KSImageNamed-Xcode-master的使用 安装该插件后,会对文件中图片进行智能提示. (2)VVDocumenter-Xcode-master能提供

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

前言:本篇讲解,在前篇iOS开发之网络编程--使用NSURLConnection实现大文件断点续传下载的基础上,使用输出流代替文件句柄实现大文件断点续传.    在实际开发中,输入输出流用的比较少,但是用起来也是很方便的.iOS开发用到的输入输出流和在Java中的输入输出流是几乎一样的,本质也是一个意思:将网络返回的数据当做流来处理.    输入输出的理解:输入到哪里?输出到哪里?这个问题不难理解,输入输出是要站着服务器角度来思考的,下面用图来解释:    代码关键词: 1.在接收到响应头的代理

iOS开发网络篇—大文件的多线程断点下载(转)

http://www.cnblogs.com/wendingding/p/3947550.html iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件.因为实现过程较为复杂,所以下面贴出完整的代码. 实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100M,那么就在沙盒中创建一个100M的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分). 项目中用到的主要