背景:
最近项目要做上传图片功能,图片必须是高清的,所以不让压缩,上传图片是大量的,比如几百张,这个如果是用afn,将图片直接for循环加入到formData里会出现一个问题,临时变量太多,导致内存紧张,最后程序奔溃。之前写过用自动释放池解决它,但是还是效果不大。如果上传的多的话,内存还是受不了。
解决办法一适用于图片少量的如40张图片
我之前写的在这,可以看看自动释放池的方法,如果你上传图片的数量不多的话,可以用这种方法。也很简单的。链接在这里http://www.jianshu.com/p/9e84fe63d5c0
解决办法二适用于图片大量的如1000张图片
思考,为甚内存会占用那么多呢?就是因为图片一口气读到内存中了。如果咱们上传三五张图片,肯定不会出问题。如何把1000张图片分开传呢?所以必须要用到多线程的知识。创建个队列。然后挨个传。注意不要把文件存到队列里,只要先存一个文件名,执行的时候再去读取文件的内容。如果要是将image传给队列,内存还是会爆的。所以存个图片名字。一个字符串肯定没有image占用的内存大吧。上代码吧。我的图片来源于相册,所以我用的图片id。
/**
创建队列然后开始上传图片
@param LocalIdArray 获取相册的图片id数组,如果你是本地的就传递图片名字数组,或者是沙盒的文件名字数组
*/
- (void)uploadOperation:(NSArray *)LocalIdArray
{
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
self.queue = queue;
//这个就是控制同时上传几张图片的,如果是1的话就是串行队列了。我是4,是并行队列。
queue.maxConcurrentOperationCount = 4;
for (int i = 0; i<LocalIdArray.count; i++)
{
//加上自动释放池,及时的释放临时变量,防止内存奔溃
@autoreleasepool {
NSString *imageName = [NSString stringWithFormat:@"up_%d.jpg",i];
WS(weakSelf)
//创建一个任务
NSBlockOperation *uploadOperation = [NSBlockOperation blockOperationWithBlock:^{
[weakSelf uploadTaskWithLocalId:LocalIdArray[i] imageount:LocalIdArray.count imageName:imageName];
}];
//将任务加入到队列中
[queue addOperation:uploadOperation];
}
}
}
/**
开始上传单张图片
@param LocalId 图片的id
@param count 一共上传多少张图片
@param imageName 图片的名称
*/
- (void)uploadTaskWithLocalId:(NSString *)LocalId imageount:(NSInteger)count imageName:(NSString *)imageName
{
//通过图片的id转化为image,如果是图片名字或者是沙盒图片文件名字那更简单了。
CustomAlbumTool *customAlbumTool = [CustomAlbumTool sharedCustomAlbumTool];
PHFetchResult<PHAsset *> *upAssetArr = [PHAsset fetchAssetsWithLocalIdentifiers:@[LocalId] options:nil];
PHAsset *asset = [upAssetArr firstObject];
UIImage *image = [customAlbumTool getImageWithAsset:asset targetSize:PHImageManagerMaximumSize];
//afn上传的参数
NSMutableDictionary *dic = [NSMutableDictionary dictionary];
dic[@"xxx"] = [UserDataCenter xxx];
dic[@"xxx"] = self.xxx;
//因为afn上传是异步执行的所以创建一个信号量。就是为了让一个任务完全的执行完毕后才执行下一个任务。加信号量就是为了把afn异步转化为同步。如果不转化的话。queue.maxConcurrentOperationCount = 1,也没办法做到队列内同步。
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
WS(weakSelf)
[SWAYNetWorking uploadWithUrl:uploadModelUrl parameters:dic constructingBodyWithBlock:^(id<AFMultipartFormData> formData) {
NSData *data = UIImageJPEGRepresentation(image, 1.0);
[formData appendPartWithFileData:data name:@"file" fileName:imageName mimeType:@"image/jpeg"];
} withProgress:^(NSProgress *uploadProgress) {
} success:^(id responseObject) {
//图片成功了让信号量加1
dispatch_semaphore_signal(semaphore);
} failure:^(NSError *error) {
//图片传失败了让信号量加1
dispatch_semaphore_signal(semaphore);
}];
//信号量等待。DISPATCH_TIME_FOREVER 永远等到吧。
dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
}
监听全部成功刷新ui,可以定义个int型变量,上传成功一张图片加1。如果等于总的图片数量就相当于上传完成了,那么就刷新UI吧。也可以创建个上传成功刷新UI的任务。添加依赖。在任务里执行刷新UI。
/**
创建队列然后开始上传图片
@param LocalIdArray 获取相册的图片id数组,如果你是本地的就传递图片名字数组,或者是沙盒的文件名字数组
*/
- (void)uploadOperation:(NSArray *)LocalIdArray
{
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
self.queue = queue;
//这个就是控制同时上传几张图片的,如果是1的话就是串行队列了。我是4,是并行队列。
queue.maxConcurrentOperationCount = 4;
NSBlockOperation *completionOperation = [NSBlockOperation blockOperationWithBlock:^{
[[NSOperationQueue mainQueue] addOperationWithBlock:^{ // 回到主线程执行,方便更新 UI 等
}];
}];
for (int i = 0; i<LocalIdArray.count; i++)
{
//加上自动释放池,及时的释放临时变量,防止内存奔溃
@autoreleasepool {
NSString *imageName = [NSString stringWithFormat:@"up_%d.jpg",i];
WS(weakSelf)
//创建一个任务
NSBlockOperation *uploadOperation = [NSBlockOperation blockOperationWithBlock:^{
[weakSelf uploadTaskWithLocalId:LocalIdArray[i] imageount:LocalIdArray.count imageName:imageName];
}];
//添加依赖。
[completionOperation addDependency:uploadOperation];
//将任务加入到队列中
[queue addOperation:uploadOperation];
}
}
//将刷新UI的任务加入队列,当所有的上传任务结束才会调用completionOperation。
[queue addOperation:completionOperation];
}
如果要是你的业务是不能让一张图片传递失败,那么当有一张图没有传成功的话就直接取消所有任务就行了。
[weakSelf.queue cancelAllOperations];
作者:王银博
链接:http://www.jianshu.com/p/5162df747879
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。