《編寫高質(zhì)量iOS與OS X代碼的52個(gè)有效方法》--第六章 第38條
(ps:此乃讀書筆記娇澎,加深記憶,僅供大家參考)
第38條:為常用的塊類型創(chuàng)建typedef
每個(gè)塊都具備其“固有類型”(inherent type)音瓷,因而可將其賦值給適當(dāng)類型的變量锦庸。這個(gè)類型由塊所接受的參數(shù)及其返回值組成敬锐。
如果想把塊賦給變量,則需注意其類型剩盒。變量類型及相關(guān)賦值語(yǔ)句如下:
int someInt = 0;
int (^variableName)(BOOL flag, int value) = ^(BOOL flag, int value)
{
return someInt;
};
這個(gè)類型似乎和普通的類型大不相同谷婆,然而如果習(xí)慣函數(shù)指針的話,那么看上去就會(huì)覺得眼熟了。塊類型的語(yǔ)法結(jié)構(gòu)如下:
return_type (^block_name)(parameters)
與其他類型的變量不同纪挎,在定義塊變量時(shí)期贫,要把變量名放在類型之中,而不要放在右側(cè)异袄。這種語(yǔ)法非常難記通砍,也非常難讀。鑒于此烤蜕,我們應(yīng)該為常用的塊類型起個(gè)別名封孙,尤其是打算把代碼發(fā)布成API供他人使用時(shí),更應(yīng)該這樣做讽营。開發(fā)者可以起個(gè)更為易讀的名字來(lái)表示塊的用途虎忌,而把塊的類型隱藏在其后面。
為了隱藏復(fù)雜的塊類型橱鹏,需要用到C語(yǔ)言中名為“類型定義”(type definition)的特性呐籽。typedef關(guān)鍵字用于給類型起個(gè)易讀的別名:
typedef int (^EOCSomeBlock)(BOOL flag, int value);
聲明變量時(shí),要把名稱放在類型中間蚀瘸,并在前面加上“^”符號(hào),而定義新類型時(shí)也得這么做庶橱。此后贮勃,不用再以復(fù)雜的塊類型來(lái)創(chuàng)建變量了,直接使用新類型即可:
EOCSomeBlock block = ^(BOOL flag, int value)
{
return someInt;
};
與定義其他變量時(shí)一樣苏章,變量類型在左邊寂嘉,變量名在右邊。類里面有些方法可能需要用塊來(lái)做參數(shù)枫绅,比如執(zhí)行異步任務(wù)時(shí)所用的“completion handler”(任務(wù)完成后所執(zhí)行的處理程序)參數(shù)就是塊泉孩,凡遇到這種情況,都可以通過定義別名使代碼變得更為易讀并淋。
注意寓搬,定義方法參數(shù)所用的塊類型語(yǔ)法,又和定義變量時(shí)不同县耽。若能把方法簽名中的參數(shù)類型寫成一個(gè)詞句喷,那讀起來(lái)就順口多了。于是可以給參數(shù)類型起個(gè)別名兔毙,然后使用此名稱來(lái)定義:
typedef void (^EOCCompletionHandler)(NSData *data, NSError *error);
- (void)startWithCompletionHandler:(EOCCompletionHandler)completion;
當(dāng)前唾琼,優(yōu)秀的集成開發(fā)環(huán)境(Integrated Development Environment, IDE)都可以自動(dòng)把類型定義展開澎剥,所以typedef這個(gè)功能變得很實(shí)用锡溯。
使用類型定義還有個(gè)好處,就是當(dāng)你打算重構(gòu)塊的類型簽名時(shí)會(huì)很方便。比方說(shuō)祭饭,要給原來(lái)的completion handler塊再加一個(gè)參數(shù)芜茵,用以表示完成任務(wù)所花的時(shí)間,那么只需修改類型定義語(yǔ)句即可:
typedef void (^EOCCompletionHandler)(NSData *data, NSTimeInterval duration, NSError *error);
修改之后甜癞,凡是使用了這個(gè)類型定義的地方夕晓,比如方法簽名處,都會(huì)無(wú)法編譯悠咱,而且報(bào)的是同一種錯(cuò)誤蒸辆,于是開發(fā)者可據(jù)此逐個(gè)修復(fù)。
最好在使用塊類型的類中定義這些typedef析既,而且還應(yīng)該把這個(gè)類的名字加在由typedef所定義的新類型名前面躬贡,這樣可以闡明塊的用途。還可以用typedef給同一個(gè)塊簽名類型創(chuàng)建數(shù)個(gè)別名眼坏。
Mac OS X與iOS的Accounts框架就是個(gè)例子拂玻。在該框架中可以找打下面這兩個(gè)類型定義語(yǔ)句:
typedef void(^ACAccountStoreSaveCompletionHandler)(BOOL success, NSError *error);
typedef void(^ACAccountStoreRemoveCompletionHandler)(BOOL success, NSError *error);
typedef void(^ACAccountStoreRequestAccessCompletionHandler)(BOOL granted, NSError *error);
類型定義的簽名相同,但用在不同的地方宰译。開發(fā)者看到類型別名及簽名中的參數(shù)之后檐蚜,很容易就能理解此類型的用途。
與此相似沿侈,如果有好幾個(gè)類都要執(zhí)行相似但各有區(qū)別的異步任務(wù)闯第,而這幾個(gè)類又不能放入同一個(gè)繼承體系,那么缀拭,每個(gè)類就應(yīng)該有自己的completion handler類型咳短。這幾個(gè)completion handler的簽名也許完全相同,但最好還是在每個(gè)類里都各自定義一個(gè)別名蛛淋,而不要共用同一個(gè)名稱咙好。反之,若這些類能納入同一個(gè)繼承中褐荷,則應(yīng)該將類型定義的語(yǔ)句放在超類中勾效,以供各子類使用。
要點(diǎn)
- 以typedef重新定義塊類型诚卸,可令塊變量用起來(lái)更加簡(jiǎn)單葵第。
- 定義新類型時(shí)應(yīng)遵從現(xiàn)有的命名習(xí)慣,勿使其名稱與別的類型相沖突合溺。
- 不妨為同一塊簽名定義多個(gè)類型別名卒密。如果要重構(gòu)代碼使用了塊類型的某個(gè)別名,那么只需修改相應(yīng)typedef中的塊簽名即可棠赛,無(wú)須改動(dòng)其他typedef哮奇。