問(wèn)題引出
在實(shí)際開發(fā)中舱痘,APP首頁(yè)的數(shù)據(jù)經(jīng)常是由多個(gè)請(qǐng)求返回的,在下拉刷新內(nèi)容的時(shí)候是需要在接收到這幾個(gè)并發(fā)請(qǐng)求的返回?cái)?shù)據(jù)之后才把下拉刷新的頭部結(jié)束刷新,恢復(fù)正常狀態(tài)录语,這個(gè)時(shí)候就需要實(shí)現(xiàn)線程的同步了。
解決方法
GCD中處理多個(gè)線程同步問(wèn)題的函數(shù)有3個(gè):dispatch_group_t 禾乘、dispatch_barrier_async澎埠、dispach_semaphore,各有各的適用環(huán)境:
1.> dispatch_group_t比較適合同步多個(gè)線程的操作, 在各個(gè)線程的任務(wù)完成后始藕,使用dispatch_group_notify函數(shù)接收組隊(duì)列通知即可.
2.> dispatch_barrier比較適合對(duì)于同一個(gè)并發(fā)隊(duì)列蒲稳,使隊(duì)列中前幾個(gè)任務(wù)完成后再執(zhí)行后面的并發(fā)任務(wù),有點(diǎn)類似于NSOperation里面的設(shè)置依賴.
3.> dispach_semaphore信號(hào)量是比較適合用來(lái)控制最大并發(fā)任務(wù)數(shù)的伍派,信號(hào)量相當(dāng)于可以當(dāng)前可以執(zhí)行的任務(wù)數(shù)弟塞,使用dispatch_semaphore_wait函數(shù)來(lái)等待信號(hào)量,當(dāng)信號(hào)數(shù)大于0時(shí)才會(huì)開始執(zhí)行并且使信號(hào)量減1.
所以開始提出的問(wèn)題使用組隊(duì)列能更方便的解決.
實(shí)際代碼
// 創(chuàng)建線程組,以便同步這3個(gè)請(qǐng)求
self.groupQueue = dispatch_group_create();
[self sendRequest1];
[self sendRequest2];
[self sendRequest2];
dispatch_group_notify(self.groupQueue, dispatch_get_main_queue(), ^{
DLOG(@"結(jié)束了3個(gè)請(qǐng)求");
self.groupQueue = nil;
[self.tableView.mj_header endRefreshing];
});
實(shí)際發(fā)送請(qǐng)求時(shí)調(diào)用的函數(shù) dispatch_group_enter(self.groupQueue)拙已、dispatch_group_leave(self.groupQueue)
#pragma mark - 輪播廣告相關(guān)
- (void) sendRequest1 {
NSMutableDictionary *params = [NSMutableDictionary dictionary];
@weakify(self);
if (self.groupQueue) {
dispatch_group_enter(self.groupQueue);
}
[[NetworkingManager new] sendHttp:CM020 content: params controller:nil animate:NO completion:^( NSDictionary *backContentDict) {
@strongify(self);
if (self.groupQueue) {
dispatch_group_leave(self.groupQueue);
}
// do some thing
} error:^(NSError *error) {
@strongify(self);
if (self.groupQueue) {
dispatch_group_leave(self.groupQueue);
}
// do some thing
}];
}