最近的需求中有一个类似发朋友圈这样的功能,其中涉及到了多图片上传,第一想法是肯定不能同步去上传,要开异步线程去实现保证效率。想来想去还是选择了GCD中的调度组来实现,上代码
我把AFN内部的上传图片方法封装在了网络工具类内
// 这里是封装的AFN的方法
- (void)uploadUrlWithData:(NSData *)data callback:(void(^)(id response, NSError *error))callback{
[self POST:@"接口URL" parameters:nil constructingBodyWithBlock:^(id<AFMultipartFormData> _Nonnull formData) {
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
formatter.dateFormat = @"yyyyMMddHHmmss";
NSString *str = [formatter stringFromDate:[NSDate date]];
NSString *fileName = [NSString stringWithFormat:@"%@%d.jpg", str, arc4random() % 100];
[formData appendPartWithFileData:data name:@"files" fileName:fileName mimeType:@"image/jpg"];
} progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:responseObject options:NSJSONReadingAllowFragments error:nil];
callback(dict,err);
}else{
callback(dict,nil);
}
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
callback(nil, error);
}];
}
- (void)uploadImage{
NSMutableArray *urlArray = [NSMutableArray arrayWithCapacity:10];
dispatch_group_t group = dispatch_group_create();
for (NSInteger i = 0 ; i < self.pictures.count; i++) {
dispatch_group_enter(group);
NSData *data = UIImageJPEGRepresentation(self.pictures[i], 0.5);
[[NetworkTools shareTools] uploadUrlWithData:data callback:^(id response, NSError *error) {
if (!error) {
NSArray *array = response;
if (array.count <= 0) {
return ;
}
NSString *url = array.firstObject;
[urlArray addObject:url];
dispatch_group_leave(group);
}else{
dispatch_group_leave(group);
return ;
}
}];
}
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 所有任务都完成后会走这个回调,在这里操作最后的数组即可
});
}
然后测试后发现最后上传成功的图片顺序并不是当时选择的图片的顺序,因为虽然在调度组内部加入了所有的任务,但是每个任务都是异步的,并不一定哪张图片先上传完成,所以还要继续改造。
整个网络请求都是包在for循环内部的,请求内回调是一个block,block可以保存变量,那这就好办了
- (void)uploadImage{
NSMutableArray *urlArray = [NSMutableArray arrayWithCapacity:10];
// 先在装载URL的数组中用 [NSNull null] 来占位
for (int i = 0; i < self.pictures.count; i++) {
[urlArray addObject:[NSNull null]];
}
dispatch_group_t group = dispatch_group_create();
for (NSInteger i = 0 ; i < self.pictures.count; i++) {
dispatch_group_enter(group);
NSData *data = UIImageJPEGRepresentation(self.pictures[i], 0.5);
[[NetworkTools shareTools] uploadUrlWithData:data callback:^(id response, NSError *error) {
if (!error) {
NSArray *array = response;
if (array.count <= 0) {
return ;
}
NSString *url = array.firstObject;
// 然后在回调后替换原本数组中的 [NSNull null] 对象
// 因为 NSMutableArray 是非线程安全的,如果多个线程同时操作数组就会crash
// 所以此处加上一个锁,保证同一时间只有一个线程能操作数组
@synchronized(self){
[urlArray replaceObjectAtIndex:i withObject:url];
}
dispatch_group_leave(group);
}else{
dispatch_group_leave(group);
return ;
}
}];
}
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 所有任务都完成后会走这个回调,在这里操作最后的数组即可
});
}
现在再测试就没有问题了。