小文件下载
如果文件过小,下载方式有很多种:
- 使用NSData的 + (id)dataWithContentsOfURL:(NSURL *)url;
- (NSURLConnection-sendAsync),使用NSURLConnection发送异步请求下载文件资源
- (NSURLConnection-delegate),使用NSURLConnection设置代理发送异步请求的方式下载文件
- 如果是下载图片,可以直接采用SDWebImage下载
大文件下载
实现思路,边接收数据边写文件以解决内存越来越大的问题
采用NSURLConnection-delegate方式下载
- 在connection: didReceiveResponse:方法中通过响应头获取要下载文件的总大小,获取文件管理者、拼接文件全路径、在该路径下创建一个空的文件夹,用来当接收到服务器返回数据的时候往该文件中写入数据
- 在connection: didReceiveData: 方法中创建一个用来向文件中写数据的文件句柄 (设置写数据的位置、写入数据、计算当前文件的下载进度)
注意当下载完成之后,该文件句柄需要关闭,调用closeFile方法
大文件断点续传
实现思路:在下载文件的时候不再是整块的从头开始下载,而是看当前文件已经下载到哪个地方,然后从该地方接着往后面下载。可以通过在请求对象中设置请求头实现。
- 创建可变请求对象,设置下载文件的某一部分,只要设置HTTP请求头的Range属性, 就可以实现从指定位置开始下载
NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.currentLength];
[request setValue:range forHTTPHeaderField:@"Range"];
- 注意点(下载进度并判断是否需要重新创建文件)
- 获得当前要下载文件的总大小
注意点: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类型
- 直接对该对象发送一个异步网络请求,在响应头中通过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;
}
- 通过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__);
}