当前位置: 首页 > 编程笔记 >

iOS开发中文件的上传和下载功能的基本实现

东门清夷
2023-03-14
本文向大家介绍iOS开发中文件的上传和下载功能的基本实现,包括了iOS开发中文件的上传和下载功能的基本实现的使用技巧和注意事项,需要的朋友参考一下

文件的上传

说明:文件上传使用的时POST请求,通常把要上传的数据保存在请求体中。本文介绍如何不借助第三方框架实现iOS开发中得文件上传。

  由于过程较为复杂,因此本文只贴出部分关键代码。

主控制器的关键代码:


YYViewController.m

#import "YYViewController.h"

#define YYEncode(str) [str dataUsingEncoding:NSUTF8StringEncoding]

@interface YYViewController ()

@end



@implementation YYViewController

- (void)viewDidLoad {     [super viewDidLoad];     // Do any additional setup after loading the view, typically from a nib. }

- (void)upload:(NSString *)name filename:(NSString *)filename mimeType:(NSString *)mimeType data:(NSData *)data parmas:(NSDictionary *)params {     // 文件上传     NSURL *url = [NSURL URLWithString:@"http://192.168.1.200:8080/YYServer/upload"];     NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];     request.HTTPMethod = @"POST";         // 设置请求体     NSMutableData *body = [NSMutableData data];         /***************文件参数***************/     // 参数开始的标志     [body appendData:YYEncode(@"--YY\r\n")];     // name : 指定参数名(必须跟服务器端保持一致)     // filename : 文件名     NSString *disposition = [NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, filename];     [body appendData:YYEncode(disposition)];     NSString *type = [NSString stringWithFormat:@"Content-Type: %@\r\n", mimeType];     [body appendData:YYEncode(type)];         [body appendData:YYEncode(@"\r\n")];     [body appendData:data];     [body appendData:YYEncode(@"\r\n")];         /***************普通参数***************/     [params enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {         // 参数开始的标志         [body appendData:YYEncode(@"--YY\r\n")];         NSString *disposition = [NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"\r\n", key];         [body appendData:YYEncode(disposition)];

        [body appendData:YYEncode(@"\r\n")];         [body appendData:YYEncode(obj)];         [body appendData:YYEncode(@"\r\n")];     }];         /***************参数结束***************/     // YY--\r\n     [body appendData:YYEncode(@"--YY--\r\n")];     request.HTTPBody = body;         // 设置请求头     // 请求体的长度     [request setValue:[NSString stringWithFormat:@"%zd", body.length] forHTTPHeaderField:@"Content-Length"];     // 声明这个POST请求是个文件上传     [request setValue:@"multipart/form-data; boundary=YY" forHTTPHeaderField:@"Content-Type"];         // 发送请求     [NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {         if (data) {             NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil];             NSLog(@"%@", dict);         } else {             NSLog(@"上传失败");         }     }]; }

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {     // Socket 实现断点上传         //apache-tomcat-6.0.41/conf/web.xml 查找 文件的 mimeType //    UIImage *image = [UIImage imageNamed:@"test"]; //    NSData *filedata = UIImagePNGRepresentation(image); //    [self upload:@"file" filename:@"test.png" mimeType:@"image/png" data:filedata parmas:@{@"username" : @"123"}];         // 给本地文件发送一个请求     NSURL *fileurl = [[NSBundle mainBundle] URLForResource:@"itcast.txt" withExtension:nil];     NSURLRequest *request = [NSURLRequest requestWithURL:fileurl];     NSURLResponse *repsonse = nil;     NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:&repsonse error:nil];         // 得到mimeType     NSLog(@"%@", repsonse.MIMEType);     [self upload:@"file" filename:@"itcast.txt" mimeType:repsonse.MIMEType data:data parmas:@{                                                                                               @"username" : @"999",                                                                                               @"type" : @"XML"}]; }

@end


补充说明:

文件上传请求数据格式

部分文件的MIMEType

多线程断点下载
说明:本文介绍多线程断点下载。项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件。因为实现过程较为复杂,所以下面贴出完整的代码。

实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100M,那么就在沙盒中创建一个100M的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分)。

项目中用到的主要类如下:

完成的实现代码如下:

主控制器中的代码:


#import "YYViewController.h"

#import "YYFileMultiDownloader.h"

@interface YYViewController () @property (nonatomic, strong) YYFileMultiDownloader *fileMultiDownloader; @end



@implementation YYViewController

-  (YYFileMultiDownloader *)fileMultiDownloader

{

    if (!_fileMultiDownloader) {

        _fileMultiDownloader = [[YYFileMultiDownloader alloc] init];

        // 需要下载的文件远程URL

        _fileMultiDownloader.url = @"http://192.168.1.200:8080/MJServer/resources/jre.zip";

        // 文件保存到什么地方

        NSString *caches = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];

        NSString *filepath = [caches stringByAppendingPathComponent:@"jre.zip"];

        _fileMultiDownloader.destPath = filepath;

    }

    return _fileMultiDownloader;

}

- (void)viewDidLoad {     [super viewDidLoad];     }

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {     [self.fileMultiDownloader start]; }

@end


自定义一个基类

YYFileDownloader.h文件

#import <Foundation/Foundation.h>

@interface YYFileDownloader : NSObject {     BOOL _downloading; } /**  * 所需要下载文件的远程URL(连接服务器的路径)  */ @property (nonatomic, copy) NSString *url; /**  * 文件的存储路径(文件下载到什么地方)  */ @property (nonatomic, copy) NSString *destPath;

/**  * 是否正在下载(有没有在下载, 只有下载器内部才知道)  */ @property (nonatomic, readonly, getter = isDownloading) BOOL downloading;

/**  * 用来监听下载进度  */ @property (nonatomic, copy) void (^progressHandler)(double progress);

/**  * 开始(恢复)下载  */ - (void)start;

/**  * 暂停下载  */ - (void)pause; @end


YYFileDownloader.m文件

#import "YYFileDownloader.h"

@implementation YYFileDownloader @end


下载器类继承自YYFileDownloader这个类

YYFileSingDownloader.h文件


#import "YYFileDownloader.h"

@interface YYFileSingleDownloader : YYFileDownloader /**  *  开始的位置  */ @property (nonatomic, assign) long long begin; /**  *  结束的位置  */ @property (nonatomic, assign) long long end; @end


YYFileSingDownloader.m文件

#import "YYFileSingleDownloader.h"

@interface YYFileSingleDownloader() <NSURLConnectionDataDelegate>

/**

 * 连接对象

 */

@property (nonatomic, strong) NSURLConnection *conn;

/**  *  写数据的文件句柄  */ @property (nonatomic, strong) NSFileHandle *writeHandle; /**  *  当前已下载数据的长度  */ @property (nonatomic, assign) long long currentLength; @end



@implementation YYFileSingleDownloader

- (NSFileHandle *)writeHandle {     if (!_writeHandle) {         _writeHandle = [NSFileHandle fileHandleForWritingAtPath:self.destPath];     }     return _writeHandle; }

/**  * 开始(恢复)下载  */ - (void)start {     NSURL *url = [NSURL URLWithString:self.url];     // 默认就是GET请求     NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];     // 设置请求头信息     NSString *value = [NSString stringWithFormat:@"bytes=%lld-%lld", self.begin + self.currentLength, self.end];     [request setValue:value forHTTPHeaderField:@"Range"];     self.conn = [NSURLConnection connectionWithRequest:request delegate:self];         _downloading = YES; }

/**  * 暂停下载  */ - (void)pause {     [self.conn cancel];     self.conn = nil;         _downloading = NO; }

#pragma mark - NSURLConnectionDataDelegate 代理方法 /**  *  1. 当接受到服务器的响应(连通了服务器)就会调用  */ - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {     }

/**  *  2. 当接受到服务器的数据就会调用(可能会被调用多次, 每次调用只会传递部分数据)  */ - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {     // 移动到文件的尾部     [self.writeHandle seekToFileOffset:self.begin + self.currentLength];     // 从当前移动的位置(文件尾部)开始写入数据     [self.writeHandle writeData:data];         // 累加长度     self.currentLength += data.length;         // 打印下载进度     double progress = (double)self.currentLength / (self.end - self.begin);     if (self.progressHandler) {         self.progressHandler(progress);     } }

/**  *  3. 当服务器的数据接受完毕后就会调用  */ - (void)connectionDidFinishLoading:(NSURLConnection *)connection {     // 清空属性值     self.currentLength = 0;         // 关闭连接(不再输入数据到文件中)     [self.writeHandle closeFile];     self.writeHandle = nil; }

/**  *  请求错误(失败)的时候调用(请求超时\断网\没有网, 一般指客户端错误)  */ - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {     }

@end


设计多线程下载器(利用HMFileMultiDownloader能开启多个线程同时下载一个文件)

一个多线程下载器只下载一个文件

YYFileMultiDownloader.h文件


#import "YYFileDownloader.h"

@interface YYFileMultiDownloader : YYFileDownloader   @end


YYFileMultiDownloader.m文件

#import "YYFileMultiDownloader.h"

#import "YYFileSingleDownloader.h"

#define YYMaxDownloadCount 4

@interface YYFileMultiDownloader() @property (nonatomic, strong) NSMutableArray *singleDownloaders; @property (nonatomic, assign) long long totalLength; @end



@implementation YYFileMultiDownloader

- (void)getFilesize {     NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:self.url]];     request.HTTPMethod = @"HEAD";         NSURLResponse *response = nil; #warning 这里要用异步请求     [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:nil];     self.totalLength = response.expectedContentLength; }

- (NSMutableArray *)singleDownloaders {     if (!_singleDownloaders) {         _singleDownloaders = [NSMutableArray array];                 // 获得文件大小         [self getFilesize];                 // 每条路径的下载量         long long size = 0;         if (self.totalLength % YYMaxDownloadCount == 0) {             size = self.totalLength / YYMaxDownloadCount;         } else {             size = self.totalLength / YYMaxDownloadCount + 1;         }                 // 创建N个下载器         for (int i = 0; i<YYMaxDownloadCount; i++) {             YYFileSingleDownloader *singleDownloader = [[YYFileSingleDownloader alloc] init];             singleDownloader.url = self.url;             singleDownloader.destPath = self.destPath;             singleDownloader.begin = i * size;             singleDownloader.end = singleDownloader.begin + size - 1;             singleDownloader.progressHandler = ^(double progress){                 NSLog(@"%d --- %f", i, progress);             };             [_singleDownloaders addObject:singleDownloader];         }                 // 创建一个跟服务器文件等大小的临时文件         [[NSFileManager defaultManager] createFileAtPath:self.destPath contents:nil attributes:nil];                 // 让self.destPath文件的长度是self.totalLengt         NSFileHandle *handle = [NSFileHandle fileHandleForWritingAtPath:self.destPath];         [handle truncateFileAtOffset:self.totalLength];     }     return _singleDownloaders; }

/**  * 开始(恢复)下载  */ - (void)start {     [self.singleDownloaders makeObjectsPerformSelector:@selector(start)];         _downloading = YES; }

/**  * 暂停下载  */ - (void)pause {     [self.singleDownloaders makeObjectsPerformSelector:@selector(pause)];     _downloading = NO; }

@end


补充说明:如何获得将要下载的文件的大小?

 类似资料:
  • 本文向大家介绍iOS开发实现下载器的基本功能(1),包括了iOS开发实现下载器的基本功能(1)的使用技巧和注意事项,需要的朋友参考一下 今天,做了一个下载器的Demo,即从本地配置的Apache服务器上,下载指定的文件。这次,我们下载服务器根目录下的html.mp4文件。 按照惯例,我们先创建一个URL对象和请求。  NSURL *url = [NSURL URLWithString:@"http

  • 本文向大家介绍springboot实现文件上传和下载功能,包括了springboot实现文件上传和下载功能的使用技巧和注意事项,需要的朋友参考一下 spring boot 引入”约定大于配置“的概念,实现自动配置,节约了开发人员的开发成本,并且凭借其微服务架构的方式和较少的配置,一出来就占据大片开发人员的芳心。大部分的配置从开发人员可见变成了相对透明了,要想进一步熟悉还需要关注源码。 1.文件上传

  • 本文向大家介绍Retrofit+Rxjava实现文件上传和下载功能,包括了Retrofit+Rxjava实现文件上传和下载功能的使用技巧和注意事项,需要的朋友参考一下 Retrofit简介: 在Android API4.4之后,Google官方使用了square公司推出的okHttp替换了HttpClient的请求方式。后来square公司又推出了基于okHttp的网络请求框架:Retrofit。

  • 本文向大家介绍javaweb实现文件上传与下载功能,包括了javaweb实现文件上传与下载功能的使用技巧和注意事项,需要的朋友参考一下 本文实例为大家分享了javaweb文件的上传与下载,供大家参考,具体内容如下 一、文件上传 要实现Web开发的上传功能,通常需要完成两步操作:一是在Web页面中添加上传输入项;二是在Servlet中读取上传文件的数据,并保存到本地硬盘中。 上传大多数情况是通过表单

  • 本文向大家介绍c# 实现文件上传下载功能的实例代码,包括了c# 实现文件上传下载功能的实例代码的使用技巧和注意事项,需要的朋友参考一下 NuGet 安装SqlSugar 1.Model文件下新建 DbContext 类 2.建uploading实体类 3.Manager文件下建UploadingManager 窗体加载Form1_Load 1.读取到数据库字段name并赋值 2.comboBox事

  • 本文向大家介绍C#实现文件上传与下载功能实例,包括了C#实现文件上传与下载功能实例的使用技巧和注意事项,需要的朋友参考一下 最近学习了 C#实现文件上传与下载,现在分享给大家。 1、C#文件上传 创建MyUpload.htm页面,用于测试 创建UploadFile.aspx文件,在UploadFile.aspx.cs键入如下代码: 2 、C#文件下载 创建DownloadFile.aspx,在Do