最近做到一個(gè)頁(yè)面里有四個(gè)接口,每一個(gè)接口返回都有對(duì)應(yīng)的UI改變,所以要四個(gè)接口全部請(qǐng)求返回成功后,再去做UI刷新。然而由于網(wǎng)絡(luò)請(qǐng)求用的是AFN等第三方庫(kù),本身就是異步的,所以GCD的柵欄函數(shù)dispatch_barrier_async是沒(méi)有用的。
那么就讓他一個(gè)一個(gè)的執(zhí)行,等到所有都執(zhí)行完成后再去刷新UI,于是乎呢,可以在第一個(gè)網(wǎng)絡(luò)請(qǐng)求了的返回block里去做第二個(gè)網(wǎng)絡(luò)請(qǐng)求,以此類(lèi)推就可以完成最后的UI刷新。這個(gè)不失為一種比較有效的方法,但是這樣寫(xiě)的代碼真的挺丑的。。。。
如果單純的想讓這幾個(gè)網(wǎng)絡(luò)請(qǐng)求按順序執(zhí)行,先進(jìn)先出,那就是隊(duì)列啊,iOS里提供的隊(duì)列管理就是GCD這個(gè)強(qiáng)大的c語(yǔ)言實(shí)現(xiàn)的庫(kù)。GCD里面有一個(gè)信號(hào)量--dispatch_semaphore_t,
dispatch_semaphore_t
dispatch_semaphore_create(long value);創(chuàng)建一個(gè)信號(hào)量,如果value小于0的話(huà),這個(gè)信號(hào)量就是nil。
dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);可以讓信號(hào)量減1,如果信號(hào)量是0,那么他會(huì)等待信號(hào)量為非零才會(huì)執(zhí)行下一步。
dispatch_semaphore_signal(dispatch_semaphore_t dsema);可以讓信號(hào)量增加1.
通過(guò)這上面的方法,我們可以一開(kāi)始設(shè)計(jì)信號(hào)量是0,每一個(gè)網(wǎng)絡(luò)請(qǐng)求之前用dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);,請(qǐng)求結(jié)束回調(diào)里用dispatch_semaphore_signal(dispatch_semaphore_t dsema);這樣可以讓請(qǐng)求按照順序執(zhí)行下來(lái)。Talk is cheep,show your the code:
- (void)semaphoretest {
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
dispatch_async(dispatch_get_global_queue(0, 0), ^{
sleep(2);
NSLog(@"任務(wù)1完成---- %@", [NSThread currentThread]);
dispatch_semaphore_signal(semaphore);
});
dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
dispatch_async(dispatch_get_global_queue(0, 0), ^{
sleep(2);
NSLog(@"任務(wù)2完成---- %@", [NSThread currentThread]);
dispatch_semaphore_signal(semaphore);
});
dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
dispatch_async(dispatch_get_global_queue(0, 0), ^{
sleep(2);
NSLog(@"任務(wù)3完成---- %@", [NSThread currentThread]);
dispatch_semaphore_signal(semaphore);
});
}
打印結(jié)果如下:

很清楚的看到了他們是按照順序執(zhí)行的。
dispatch_group
最好的設(shè)計(jì)當(dāng)然是他們幾個(gè)并發(fā)執(zhí)行,到時(shí)候全部請(qǐng)求結(jié)束后再去做UI的刷新。于是乎想到了CGD的group啦!GCD就是辣么強(qiáng)大有木有!設(shè)計(jì)如下:
- (void)groupTest {
dispatch_queue_t quete = dispatch_queue_create("XIAXIAQUEUE", DISPATCH_QUEUE_CONCURRENT);
dispatch_group_t group = dispatch_group_create();
dispatch_group_enter(group);
dispatch_group_async(group, quete, ^{
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
sleep(5);
NSLog(@"網(wǎng)絡(luò)任務(wù)1完----%@", [NSThread currentThread]);
dispatch_group_leave(group);
});
sleep(2);
NSLog(@"任務(wù)1完成---%@", [NSThread currentThread]);
});
// dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
dispatch_group_enter(group);
dispatch_group_async(group, quete, ^{
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
sleep(5);
NSLog(@"網(wǎng)絡(luò)任務(wù)2完成----%@", [NSThread currentThread]);
dispatch_group_leave(group);
});
sleep(2);
NSLog(@"任務(wù)2完成---%@", [NSThread currentThread]);
});
// dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
dispatch_group_enter(group);
dispatch_group_async(group, quete, ^{
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
sleep(5);
NSLog(@"網(wǎng)絡(luò)任務(wù)3完成 --- %@", [NSThread currentThread]);
dispatch_group_leave(group);
});
sleep(2);
NSLog(@"任務(wù)3完成---- %@", [NSThread currentThread]);
});
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
NSLog(@"最后執(zhí)行 ---- %@", [NSThread currentThread]);
});
}
打印的結(jié)果如下:

首先可以發(fā)現(xiàn)其實(shí)這幾個(gè)任務(wù)都是在不同的線(xiàn)程的,但是最后執(zhí)行的肯定是最后一步!
結(jié)束語(yǔ)
作為一個(gè)程序猿,思考是最重要的,雖然現(xiàn)在網(wǎng)上很多代碼可以用拿來(lái)主義,但是如果只會(huì)拿來(lái),那只能作為一個(gè)初級(jí)的程序猿了。所以有時(shí)候多思考,多想想。