塊與大中樞派發(fā)
第 37 條:理解“塊”這一概念
void (^somBlock)() = ^ {
// Block implementation here
}
int (^addBlock)(int a, int b) = ^(int a, int b) {
return a + b;
};
int add = addBlock(2, 5); //< add = 7
塊的強(qiáng)大之處是:在聲明它的范圍里丈积,所有變量都可以為其所捕獲。這就是說(shuō)浇衬,那個(gè)范圍里的全部變量缝其,在塊里依然可用湾盗。
- 塊是 C C++ Objective-C 中得詞法閉包
- 塊可接受參數(shù),也可返回值。
- 塊可以分配在棧或zui上械荷,也可以是全局的。分配在棧上得塊可拷貝到最里虑灰,這樣的花吨瞎,就和標(biāo)準(zhǔn)的 Objective-C 對(duì)象一樣,具備引用計(jì)數(shù)了穆咐。
第 38 條:為常用的塊類(lèi)型創(chuàng)建 typedef
typedef int (^EOCSomeBlock)(BOOL flag, int value);
直接使用新類(lèi)型
EOCSomeBlock block = ^(BOOL flag, int value) {
//Implementation
} - 以 typedef 重新定義塊類(lèi)型颤诀,可令塊變量用起來(lái)更加簡(jiǎn)單。
- 定義新類(lèi)型時(shí)應(yīng)遵從現(xiàn)有的命名習(xí)慣对湃,勿使其名稱(chēng)與別的類(lèi)型相沖突崖叫。
- 不妨為同一個(gè)塊簽名定義多個(gè)類(lèi)型別名。如果要重構(gòu)的代碼使用了塊類(lèi)型的某個(gè)別名拍柒,那么只需要修改相應(yīng) typedef 中的塊簽名即可心傀,無(wú)須改動(dòng)其他 typedef。
第 39 條:用 handle 塊降低代碼分散程度 - 在創(chuàng)建對(duì)象時(shí)拆讯,可以使用內(nèi)聯(lián)的 handler 塊將相關(guān)業(yè)務(wù)邏輯一并聲明脂男。
- 在有多個(gè)實(shí)例需要監(jiān)控時(shí)养叛,如果采用委托模式,那么經(jīng)常需要根據(jù)傳入的對(duì)象來(lái)切換宰翅,而若改用 handler 塊來(lái)實(shí)現(xiàn)弃甥,則可直接將塊與相關(guān)對(duì)象放在一起。
- 設(shè)計(jì) API 時(shí)如果用到了 handler 塊汁讼,那么可以增加一個(gè)參數(shù)淆攻,使調(diào)用者可通過(guò)此參數(shù)來(lái)決定應(yīng)該把塊安排在哪個(gè)隊(duì)列上執(zhí)行。
第 40 條:用塊引用其所屬對(duì)象時(shí)不要出現(xiàn)保留環(huán) - 如果塊所捕獲的對(duì)象直接或間接地保留了塊本身嘿架,那么就得當(dāng)心保留環(huán)問(wèn)題
- 一定要找個(gè)適當(dāng)?shù)貢r(shí)機(jī)解除保留環(huán)瓶珊,而不能把責(zé)任推給 API的調(diào)用者
第 41 條:多用派發(fā)隊(duì)列,少用同步鎖
- (void)synchronizedMethod {
- @synchronized(self) {
- //safe
- }
- }
- 派發(fā)隊(duì)列可用來(lái)表述同步語(yǔ)義眶明,這種做法要比使用@synchronized 塊或 NSLock 對(duì)象更簡(jiǎn)單艰毒。
- 將同步與異步派發(fā)結(jié)合起來(lái),可以實(shí)現(xiàn)與普通加鎖機(jī)制一樣的同步行為搜囱,這么做卻不會(huì)阻塞執(zhí)行異步派發(fā)的線程丑瞧。
- 使用同步隊(duì)列及柵欄塊,可以令同步行為更加高效蜀肘。
第 42 條:多用 GCD绊汹,少用 performSelector 系列方法
// Using performSelector:withObject:afterDelay:
// Using dispatch_after
dispatch_time_t time = dispatch_time(DISPATCH_TIME_NOW,(int64_t)(5.0 * NSEC_PER_SEC));
dispatch_after (time, dispatch_get_main_queue(), ^(void) {
[self doSomething];
});
把任務(wù)放在主線程上執(zhí)行,也可以有下面兩種方式扮宠,而我們還是應(yīng)該優(yōu)選后者:
// Using performSelectorOnMainThread:withObject:waitUntilDone:
//Using dispatch_async
//(or if waitUntilDone is YES, then dispatch_sync)
dispatch_async(dispatch_get_main_queue(), ^{
[self doSomething];
});
- performSelector 系列方法在內(nèi)存管理方面容易有疏失西乖。它無(wú)法確定將要執(zhí)行的選擇子具體是什么,因而 ARC 編譯器也就無(wú)法插入適當(dāng)?shù)貎?nèi)存管理方法坛增。
- performSelector 系列方法所能處理的選擇子太過(guò)局限了获雕,選擇子的返回值類(lèi)型及發(fā)送給方法的參數(shù)個(gè)數(shù)都受到限制。
- 如果想把任務(wù)放在另一個(gè)線程上執(zhí)行收捣,那么最好不要用 performSelector 系列方法届案,而是應(yīng)該把任務(wù)封裝到塊里,然后調(diào)用GCD的相關(guān)方法來(lái)實(shí)現(xiàn)罢艾。
第 43 條:掌握 GCD 及操作隊(duì)列的使用時(shí)機(jī)
只會(huì)執(zhí)行一次的代碼
dispatch_once
在執(zhí)行后臺(tái)任務(wù)時(shí)楣颠,GCD 并不一定是最佳方式。還有一種技術(shù)叫做 NSOperationQueue咐蚯,它雖然與 GCD 不同童漩,但是卻與之相關(guān),開(kāi)發(fā)者可以把操作以 NSOperation 子類(lèi)的形式放在隊(duì)列中春锋,而這些操作也能夠并發(fā)執(zhí)行矫膨。其與 GCD 派發(fā)隊(duì)列有相似之處, - 在解決多線程與任務(wù)管理問(wèn)題時(shí),派發(fā)隊(duì)列并非唯一方案豆拨。
- 操作隊(duì)列提供了一套高層的 Objective-C API直奋,能實(shí)現(xiàn)純 GCD 所具備的絕大部分功能,而且還能完成一些更為復(fù)雜的操作施禾,那些操作若改用 GCD 來(lái)實(shí)現(xiàn),則需另外編寫(xiě)代碼搁胆。
第 44 條:通過(guò) Dispatch Group 機(jī)制弥搞,根據(jù)系統(tǒng)資源狀況來(lái)執(zhí)行任務(wù)
創(chuàng)建 dispatch group:
dispatch_group_t dispatch_group_create();
void dispatch_group_async(dispatch_group_t group, dispatch_queue_t queue, dispatch_block_t block);
void dispatch_group_enter(dispatch_group_t group);
void dispatch_group_leave(dispatch_group_t group);
collection - 一系列任務(wù)可歸入一個(gè) dispatch group 之中。開(kāi)發(fā)者可以在這組任務(wù)執(zhí)行完畢時(shí)獲得通知渠旁。
- 通過(guò) dispatch group攀例,可以在并發(fā)式派發(fā)隊(duì)列里同時(shí)執(zhí)行多項(xiàng)任務(wù)。此時(shí) GCD 會(huì)根據(jù)系統(tǒng)資源狀況來(lái)調(diào)度這些并發(fā)執(zhí)行的任務(wù)顾腊。開(kāi)發(fā)者若自己來(lái)實(shí)現(xiàn)此功能粤铭,則需編寫(xiě)大量代碼。
第 45 條:使用 dispatch_once 來(lái)執(zhí)行只需運(yùn)行一次的線程安全代碼
單例模式
sharedInstance
GCD 引入了一項(xiàng)特性杂靶,能使單例實(shí)現(xiàn)起來(lái)更為容易梆惯。所用的函數(shù)是:
void dispatch_once(dispatch_once_t *token, dispatch_block_t block);
180頁(yè) 單例 - 經(jīng)常需要編寫(xiě)“只需執(zhí)行一次的線程安全代碼”。通過(guò) GCD 所提供的 dispatch_once 函數(shù)吗垮,很容易就能實(shí)現(xiàn)此功能垛吗。
- 標(biāo)記應(yīng)該聲明在 static 或 global 作用域中,這樣的話烁登,在把只需執(zhí)行一次的塊傳給 dispatch_once 函數(shù)時(shí)怯屉,傳進(jìn)去的標(biāo)記也是相同的。
第 46 條:不要使用 dispatch_get_current_queue - dispatch_get_current_queue 函數(shù)的行為常常與開(kāi)發(fā)者所預(yù)期的不同饵沧。此函數(shù)已經(jīng)廢棄锨络,只應(yīng)做調(diào)試只用。
- 由于派發(fā)隊(duì)列是按層級(jí)來(lái)組織的狼牺,所以無(wú)法單用某個(gè)隊(duì)列對(duì)象來(lái)描述“當(dāng)前隊(duì)列”這一概念羡儿。
- dispatch_get_current_queue 函數(shù)用于解決不可重入的代碼所引發(fā)的死鎖,然而能用此函數(shù)解決的問(wèn)題锁右,通常也能改用“隊(duì)列特定數(shù)據(jù)”來(lái)解決失受。