在開發(fā)過程中妓羊,經(jīng)常會(huì)遇到耗時(shí)的操作需要放到子線程中,完成后再返回到主線程中同步UI驾讲,以防阻塞主線程可婶。有時(shí)候沿癞,可能需求是要同時(shí)處理多個(gè)耗時(shí)操作,在處理完所有耗時(shí)操作后再同步UI矛渴,這時(shí)候就可以使用到GCD了椎扬。
需求1
同時(shí)下載兩張圖
需要同時(shí)下載兩張圖片,在兩張圖片都下載完成后具温,同時(shí)返回蚕涤。
使用group的大致實(shí)現(xiàn)
- (void)downloadImageSuccess:(void (^)(UIImage *image1, UIImage *image2))success {
dispatch_queue_t global_queue = dispatch_get_global_queue(0, 0);
dispatch_group_t group = dispatch_group_create();
__block UIImage *image1;
__block UIImage *image2;
dispatch_group_async(group, global_queue, ^{
NSLog(@"第一張圖片開始下載");
sleep(2);
image1 = [UIImage new];
NSLog(@"第一張圖片下載完成");
});
dispatch_group_async(group, global_queue, ^{
NSLog(@"第二張圖片開始下載");
sleep(3);
image2 = [UIImage new];
NSLog(@"第二張圖片下載完成");
});
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
NSLog(@"兩張圖片下載完成");
success(image1, image2);
});
}
需求二
下載超時(shí)時(shí)間為2秒
設(shè)置圖片下載超時(shí)時(shí)間為2秒,如果圖片在2秒后沒有全部下載完成铣猩,那么沒完成的圖片用nil返回揖铜。
代碼實(shí)現(xiàn)
使用dispatch_group_wait
函數(shù)。
- (void)downloadImageSuccess:(void (^)(UIImage *image1, UIImage *image2))success {
dispatch_queue_t global_queue = dispatch_get_global_queue(0, 0);
dispatch_group_t group = dispatch_group_create();
__block UIImage *image1;
__block UIImage *image2;
dispatch_group_async(group, global_queue, ^{
NSLog(@"第一張圖片開始下載");
sleep(2);
image1 = [UIImage new];
NSLog(@"第一張圖片下載完成");
});
dispatch_group_async(group, global_queue, ^{
NSLog(@"第二張圖片開始下載");
sleep(4);
image2 = [UIImage new];
NSLog(@"第二張圖片下載完成");
});
dispatch_async(global_queue, ^{
dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, 3 * NSEC_PER_SEC));
dispatch_async(dispatch_get_main_queue(), ^{
success(image1, image2);
});
});
}
2018-12-09 21:56:15.007222+0800 GCDDemo[12279:450516] 第一張圖片開始下載
2018-12-09 21:56:15.007222+0800 GCDDemo[12279:450524] 第二張圖片開始下載
2018-12-09 21:56:17.011723+0800 GCDDemo[12279:450516] 第一張圖片下載完成
2018-12-09 21:56:18.007809+0800 GCDDemo[12279:450477] <UIImage: 0x60000186a450>, {0, 0} (null)
2018-12-09 21:56:19.008913+0800 GCDDemo[12279:450524] 第二張圖片下載完成
注意wait一定要放在子線程中达皿,否則就阻塞主線程了天吓。
具體使用GCD完成下載時(shí)遇到的問題
問題1:異步任務(wù)無法控制
光上面的代碼贿肩,其實(shí)并不能實(shí)現(xiàn)多張圖片下載的同步功能,因?yàn)槿绻褂孟到y(tǒng)的封裝好的下載的類的話龄寞,就只能這樣用:
NSURLSession *session = [NSURLSession sharedSession];
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:[NSURL URLWithString:@"image"] completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
}];
[task resume];
如果直接把這段代碼扔到dispatch_group_async
中汰规,那么只能做到異步開始下載任務(wù),并不能做到同步任務(wù)的功能物邑。
類似這樣:
dispatch_group_async(group, global_queue, ^{
dispatch_sync(con_queue, ^{
// TODO
});
});
那么要怎么去做呢溜哮?
答案是使用dispatch_group_enter
和dispatch_group_leave
方法。
具體模擬使用:
dispatch_queue_t global_queue = dispatch_get_global_queue(0, 0);
dispatch_group_t group = dispatch_group_create();
__block UIImage *image1;
__block UIImage *image2;
dispatch_group_enter(group);
dispatch_group_async(group, global_queue, ^{
dispatch_sync(con_queue), ^{
NSLog(@"第一張圖片開始下載");
sleep(2);
image1 = [UIImage new];
NSLog(@"第一張圖片下載完成");
dispatch_group_leave(group);
});
});
dispatch_group_enter(group);
dispatch_group_async(group, global_queue, ^{
dispatch_sync(con_queue), ^{
NSLog(@"第二張圖片開始下載");
sleep(4);
image2 = [UIImage new];
NSLog(@"第二張圖片下載完成");
dispatch_group_leave(group);
});
});
dispatch_async(global_queue, ^{
dispatch_group_wait(group, dispatch_time(DISPATCH_TIME_NOW, 3 * NSEC_PER_SEC));
dispatch_async(dispatch_get_main_queue(), ^{
success(image1, image2);
});
});
2018-12-09 22:23:36.918395+0800 GCDDemo[12558:492481] 第一張圖片開始下載
2018-12-09 22:23:36.918395+0800 GCDDemo[12558:492483] 第二張圖片開始下載
2018-12-09 22:23:38.923074+0800 GCDDemo[12558:492481] 第一張圖片下載完成
2018-12-09 22:23:39.919640+0800 GCDDemo[12558:492349] <UIImage: 0x600003ffb100>, {0, 0} (null)
2018-12-09 22:23:40.923513+0800 GCDDemo[12558:492483] 第二張圖片下載完成
問題2:如何取消圖片下載任務(wù)
其實(shí)使用SDWebImage的SDWebImageDownloader
類可以直接取消下載操作拂封。
同時(shí)我看到NSURLSessionDownloadTask
類中也有cancel方法茬射,但是我沒試。冒签。。
最后钟病,我的博客地址:http://jabberyq.top/