文件的下载、上传和解压

小文件下载

如果文件过小,下载方式有很多种:

  • 使用NSData的 + (id)dataWithContentsOfURL:(NSURL *)url;
  • (NSURLConnection-sendAsync),使用NSURLConnection发送异步请求下载文件资源
  • (NSURLConnection-delegate),使用NSURLConnection设置代理发送异步请求的方式下载文件
  • 如果是下载图片,可以直接采用SDWebImage下载

大文件下载

实现思路,边接收数据边写文件以解决内存越来越大的问题
采用NSURLConnection-delegate方式下载

  1. 在connection: didReceiveResponse:方法中通过响应头获取要下载文件的总大小,获取文件管理者、拼接文件全路径、在该路径下创建一个空的文件夹,用来当接收到服务器返回数据的时候往该文件中写入数据
  2. 在connection: didReceiveData: 方法中创建一个用来向文件中写数据的文件句柄 (设置写数据的位置、写入数据、计算当前文件的下载进度)
    注意当下载完成之后,该文件句柄需要关闭,调用closeFile方法

大文件断点续传

实现思路:在下载文件的时候不再是整块的从头开始下载,而是看当前文件已经下载到哪个地方,然后从该地方接着往后面下载。可以通过在请求对象中设置请求头实现。

  1. 创建可变请求对象,设置下载文件的某一部分,只要设置HTTP请求头的Range属性, 就可以实现从指定位置开始下载
NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.currentLength];
[request setValue:range forHTTPHeaderField:@"Range"];
  1. 注意点(下载进度并判断是否需要重新创建文件)
  • 获得当前要下载文件的总大小
    注意点:res.expectedContentLength获得是本次请求要下载的文件的大小(并非是完整的文件的大小)
    因此:文件的总大小 == 本次要下载的文件大小+已经下载的文件的大小
self.totalLength = res.expectedContentLength + self.currentLength;
  • 判断当前是否已经下载过,如果当前文件已经存在,那么直接返回
if (self.currentLength >0) {
    return;
}

输出流

使用输出流也可以实现和NSFileHandle相同的功能
方法:

  • 创建一个数据输出流
 // 第一个参数:二进制的流数据要写入到哪里
 // 第二个参数:采用什么样的方式写入流数据,如果YES则表示追加,如果是NO则表示覆盖
NSOutputStream *stream = [NSOutputStream outputStreamToFileAtPath:fullPath append:YES];

// 只要调用了该方法就会往文件中写数据
// 如果文件不存在,那么会自动的创建一个
[stream open];
self.stream = stream;
  • 当接收到数据的时候写数据,使用输出流写数据
// 第一个参数:要写入的二进制数据
// 第二个参数:要写入的数据的大小
[self.stream write:data.bytes maxLength:data.length];
  • 当文件下载完毕的时候关闭输出流
// 关闭输出流
[self.stream close];
self.stream = nil;

使用多线程下载文件

思路:

  • 开启多条线程,每条线程都只下载文件的一部分(通过设置请求头中的Range来实现)
  • 创建一个和需要下载文件大小一致的文件,判断当前是哪个线程,根据当前的线程来判断下载的数据应该写入到文件中的哪个位置。(假设开5条线程来下载10M的文件,那么线程1下载0-2M,线程2下载2-4M一次类推,当接收到服务器返回的数据之后应该先判断当前线程是哪个线程,假如当前线程是线程2,那么在写数据的时候就从文件的2M位置开始写入)
  • 代码相关:使用NSFileHandle这个类的seekToFileOfSet方法,来向文件中特定的位置写入数据。
  • 技术相关
    a.每个线程通过设置请求头Range属性下载文件中的某一个部分
    b.通过NSFileHandle向文件中的指定位置写数据

文件的压缩和解压缩

1. ZipArchive

需要引入libz.dylib框架,使用需要包含Main文件,如果使用cocoaPoads来安装框架,那么会自动的配置框架的使用环境

2. SSZipArchive

需要引入libz.dylib框架

文件上传

  • 步骤:
    (1)确定请求路径
    (2)根据URL创建一个可变的请求对象
    (3)设置请求对象,修改请求方式为POST
    (4)设置请求头,告诉服务器我们将要上传文件(Content-Type)
    (5)设置请求体(在请求体中按照既定的格式拼接要上传的文件参数和非文件参数等数据)
    a. 拼接文件参数
    b. 拼接非文件参数
    c. 添加结尾标记
    (6)使用NSURLConnection sendAsync发送异步请求上传文件
    (7)解析服务器返回的数据
  • 文件上传相关代码
- (void)upload
{
  // 1.确定请求路径
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/upload"];
  // 2.创建一个可变的请求对象
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];  
  // 3.设置请求方式为POST
    request.HTTPMethod = @"POST";
  // 4.设置请求头
    NSString *filed = [NSString stringWithFormat:@"multipart/form-data; boundary=%@",Kboundary];
    [request setValue:filed forHTTPHeaderField:@"Content-Type"];
  // 5.设置请求体
    NSMutableData *data = [NSMutableData data];
  // 5.1文件参数
   
   //  --分隔符
   //  Content-Disposition:参数
   //  Content-Type:参数
   //  空行
   // 文件参数
    
    [data appendData:[[NSString stringWithFormat:@"--%@",Kboundary] dataUsingEncoding:NSUTF8StringEncoding]];
    [data appendData:KnewLine];
    [data appendData:[@"Content-Disposition: form-data; name=\"file\"; filename=\"test.png\""dataUsingEncoding:NSUTF8StringEncoding]];
    [data appendData:KnewLine];
    [data appendData:[@"Content-Type: image/png" dataUsingEncoding:NSUTF8StringEncoding]];
    [data appendData:KnewLine];
    [data appendData:KnewLine];
    [data appendData:KnewLine];
    
    UIImage *image = [UIImage imageNamed:@"test"];
    NSData *imageData = UIImagePNGRepresentation(image);
    [data appendData:imageData];
    [data appendData:KnewLine];
    
    // 5.2非文件参数
   
    // --分隔符
    // Content-Disposition:参数
    // 空行
    // 非文件参数的二进制数据
    
    [data appendData:[[NSString stringWithFormat:@"--%@",Kboundary] dataUsingEncoding:NSUTF8StringEncoding]];
    [data appendData:KnewLine];
    [data appendData:[@"Content-Disposition: form-data; name=\"username\"" dataUsingEncoding:NSUTF8StringEncoding]];
    [data appendData:KnewLine];
    [data appendData:KnewLine];
    [data appendData:KnewLine];
    
    NSData *nameData = [@"wendingding" dataUsingEncoding:NSUTF8StringEncoding];
    [data appendData:nameData];
    [data appendData:KnewLine];
    
   // 5.3结尾标识
   // --分隔符--
    [data appendData:[[NSString stringWithFormat:@"--%@--",Kboundary] dataUsingEncoding:NSUTF8StringEncoding]];
    [data appendData:KnewLine];
    
    request.HTTPBody = data;
  // 6.发送请求
    [NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse * __nullable response, NSData * __nullable data, NSError * __nullable connectionError) {
        
      // 7.解析服务器返回的数据
        NSLog(@"%@",[NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]);
    }];
}
  • 获得文件的MIMEType类型
  1. 直接对该对象发送一个异步网络请求,在响应头中通过response.MIMEType拿到文件的MIMEType类型
- (NSString *)getMIMEType
{
    NSString *filePath = @"/Users/文顶顶/Desktop/备课/其它/swift.md";
    NSURLResponse *response = nil;
    [NSURLConnection sendSynchronousRequest:[NSURLRequest requestWithURL:[NSURL fileURLWithPath:filePath]] returningResponse:&response error:nil];
    return response.MIMEType;

}
  1. 通过UTTypeCopyPreferredTagWithClass方法
    注意:需要依赖于框架MobileCoreServices
- (NSString *)mimeTypeForFileAtPath:(NSString *)path
{
    if (![[[NSFileManager alloc] init] fileExistsAtPath:path]) {
        return nil;
    }
    CFStringRef UTI = UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension, (__bridge CFStringRef)[path pathExtension], NULL);
    CFStringRef MIMEType = UTTypeCopyPreferredTagWithClass (UTI, kUTTagClassMIMEType);
    CFRelease(UTI);
    if (!MIMEType) {
        return @"application/octet-stream";
    }
    return (__bridge NSString *)(MIMEType);
}

例如:

#pragma mark--使用NSURLConnection设置代理发送异步请求的方式下载文件
-(void)connectionDelegateDownload
{
  // 1.确定请求路径
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_01.mp4"];
  // 2.创建请求对象
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
  // 3.使用NSURLConnection设置代理并发送异步请求
    [NSURLConnection connectionWithRequest:request delegate:self];
}

#pragma mark--NSURLConnectionDataDelegate
// 当接收到服务器响应的时候调用,该方法只会调用一次
-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
   // 创建一个容器,用来接收服务器返回的数据
    self.fileData = [NSMutableData data];
   // 获得当前要下载文件的总大小(通过响应头得到)
    NSHTTPURLResponse *res = (NSHTTPURLResponse *)response;
    self.totalLength = res.expectedContentLength;
    NSLog(@"%zd",self.totalLength);
   // 拿到服务器端推荐的文件名称
    self.fileName = res.suggestedFilename;
}

#pragma mark--当接收到服务器返回的数据时会调用
// 该方法可能会被调用多次
-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
   // 拼接每次下载的数据
    [self.fileData appendData:data];
   // 计算当前下载进度并刷新UI显示
    self.currentLength = self.fileData.length;
    NSLog(@"%f",1.0* self.currentLength/self.totalLength);
    self.progressView.progress = 1.0*   
    self.currentLength/self.totalLength;
}

#pragma mark--当网络请求结束之后调用
-(void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    // 文件下载完毕把接受到的文件数据写入到沙盒中保存
    // 1.确定要保存文件的全路径
    // caches文件夹路径
    NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
    NSString *fullPath = [caches stringByAppendingPathComponent:self.fileName];
   // 2.写数据到文件中
    [self.fileData writeToFile:fullPath atomically:YES];
    NSLog(@"%@",fullPath);
}

#pragma mark--当请求失败的时候调用该方法
-(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
    NSLog(@"%s",__func__);
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容