《Effective Objective-C 2.0:編寫高質(zhì)量iOS與OS X代碼的52個(gè)有效方法》閱讀筆記

前言:第一次閱讀此書大概是一年半之前,在網(wǎng)上找到電子版,也就臨時(shí)看了一晚上,之后就沒(méi)有再涉獵。八月份決定抽出半個(gè)月左右的時(shí)間認(rèn)真閱讀下這本書(但是由于這兩周一直在準(zhǔn)備公司項(xiàng)目的新版本,實(shí)際只讀完了前五章)。讀后受益匪淺,今天抽空把之前做的筆記整理一部分出來(lái),做個(gè)記錄與分享。


  • 對(duì)象所占內(nèi)存總是分配在堆空間(heap space)中,而絕不會(huì)分配在棧(stack)上,不能在棧中分配OC對(duì)象;

  • 有時(shí)會(huì)遇到定義里不含*的變量,他們可能會(huì)使用棧空間(stack space)。這些變量所保存的不是OC對(duì)象,比如CGRect,是一個(gè)結(jié)構(gòu)體;

  • 創(chuàng)建對(duì)象需要額外開(kāi)銷,例如分配和釋放內(nèi)存;

  • 如果在一個(gè)類文件中,只需要知道另一類的類名,而不需要用到該類接口的情況下,盡量使用前置聲明@class而不是#import“”引入頭文件,將引入頭文件的時(shí)機(jī)盡量延后,這樣可以減少編譯時(shí)間,也可以避免循環(huán)引用,導(dǎo)致兩個(gè)類中有一個(gè)類無(wú)法被編譯。

  • 多用字面量語(yǔ)法
    1、縮減代碼長(zhǎng)度,簡(jiǎn)便整潔
    2、數(shù)組和字典使用字面量語(yǔ)法,當(dāng)集合內(nèi)對(duì)象為空時(shí),會(huì)拋出異常,這樣會(huì)更安全

  • 多用類型常量,少用#define預(yù)處理指令
    1、 預(yù)處理指令定義的常量沒(méi)有類型信息
    2、盡量在實(shí)現(xiàn)文件而不要在聲明文件中聲明預(yù)處理指令

  • 凡是需要以按位或操作來(lái)組合的枚舉都應(yīng)該使用NS_OPTIONS定義,不需要互相組合的枚舉,應(yīng)使用NS_ENUM來(lái)定義,這兩個(gè)宏具備后向兼容能力都是用#define預(yù)處理指令來(lái)定義的。
    原因:在C++的編譯模式下,認(rèn)為按位運(yùn)算的運(yùn)算結(jié)果的數(shù)據(jù)類型應(yīng)該是枚舉的底層數(shù)據(jù)類型,而且C++不允許將這個(gè)底層類型“隱式轉(zhuǎn)換”為枚舉類型本身。如果使用NS_ENUM宏來(lái)定義組合枚舉,則有可能會(huì)報(bào)錯(cuò)

  • 在switch語(yǔ)句中,若用枚舉來(lái)定義狀態(tài)機(jī),最好不要有default分支。這樣的話,如果之后又加了一種狀態(tài),編譯器就會(huì)發(fā)出警告,提示新加入的狀態(tài)并未在switch語(yǔ)句中進(jìn)行處理,可以保證代碼的準(zhǔn)確性。否則就會(huì)自動(dòng)進(jìn)入default分支進(jìn)行處理,不會(huì)發(fā)出警告。

  • 在iOS開(kāi)發(fā)中,幾乎所有屬性都聲明為nonatomic,這樣做的歷史原因是:在iOS開(kāi)發(fā)中使用同步鎖開(kāi)銷較大,會(huì)帶來(lái)性能問(wèn)題。一般情況下并不要求屬性必須是“原子的”,因?yàn)檫@并不能保證“線程安全”,若要實(shí)現(xiàn)線程安全的操作,還需要采用更為深層的鎖定機(jī)制才行。

  • “collection”與“set”在中文里都叫做“集合”,前者是Array、Dictionary、set等數(shù)據(jù)結(jié)構(gòu)的總稱。

  • 對(duì)象等同性
    “==”操作符比較的是兩個(gè)指針本身,而不是指針?biāo)傅膶?duì)象,準(zhǔn)確率較低;
    “isEqualToString”比“isEqual”執(zhí)行速度快,后者還要進(jìn)行其他操作;
    基本數(shù)據(jù)類型判斷等同性使用:“==”
    判斷等同性優(yōu)先使用:“isEqualToString/Array/Dictionary” > "isEqual" > "=="

  • 學(xué)會(huì)使用類族模式
    系統(tǒng)框架中有許多類族,特別是collection類,如數(shù)組,字典等。使用“類族模式”的主要思想是通過(guò)繼承,隱藏基類的內(nèi)部實(shí)現(xiàn)細(xì)節(jié),開(kāi)放公共接口,方便調(diào)用,便于維護(hù);
    例如在項(xiàng)目工程中為所有包含列表視圖的viewController類創(chuàng)建父類ListViewController,通過(guò)繼承來(lái)統(tǒng)一接口,統(tǒng)一管理,簡(jiǎn)化代碼;
    在子類越來(lái)越多時(shí),類族的優(yōu)勢(shì)會(huì)越來(lái)越明顯,開(kāi)發(fā)人員無(wú)需關(guān)心其他問(wèn)題,只需要在基類的實(shí)現(xiàn)方法中增加該子類的初始化接口即可。而各種屬性定義,方法實(shí)現(xiàn)從基類的公共接口處調(diào)用;
  • 關(guān)聯(lián)對(duì)象
    設(shè)置關(guān)聯(lián)對(duì)象:objc_setAssociatedObject(<#id object#>, <#const void *key#>, <#id value#>, <#objc_AssociationPolicy policy#>)
    獲取相應(yīng)的關(guān)聯(lián)對(duì)象: objc_getAssociatedObject(<#id object#>, <#const void *key#>)
    刪除所有的關(guān)聯(lián)對(duì)象:objc_removeAssociatedObjects(<#id object#>)
    注:由于設(shè)置關(guān)聯(lián)對(duì)象時(shí)用的key是“不透明指針”,就是不局限于某種特定類型的指針,若想使兩個(gè)鍵匹配到同一個(gè)值,必須是完全相同的指針,所以在設(shè)置靜態(tài)全局變量時(shí),通常使用靜態(tài)全局變量做key,也可以使用選擇器等,保證指針相等性。
    一般只有在其他辦法行不通的情況下才會(huì)使用關(guān)聯(lián)對(duì)象,因?yàn)闉E用關(guān)聯(lián)對(duì)象可能會(huì)導(dǎo)致“保留環(huán)”,使代碼失控,程序崩潰

  • C語(yǔ)言使用“靜態(tài)綁定”,在編譯期就能決定在運(yùn)行時(shí)所應(yīng)調(diào)用的函數(shù)
    在OC中,如果向某對(duì)象傳遞消息,就會(huì)使用動(dòng)態(tài)綁定機(jī)制來(lái)決定需要調(diào)用的方法
    向某一對(duì)象發(fā)送消息,編譯器看到此消息后,會(huì)將其轉(zhuǎn)換為一條標(biāo)準(zhǔn)的C語(yǔ)言函數(shù)調(diào)用,所調(diào)用的函數(shù)為消息傳遞的核心函數(shù):
    objc_msgSend();
    iOS 8.0之后調(diào)用方式:((void (*)(id,SEL,id、、、))objc_msgSend)(self,@selector(selector),object1、、、);
    objc_msgSend函數(shù)執(zhí)行方法調(diào)用的順序:
    在接受者所在類的方法列表中,查找與選擇器名稱相同的方法,進(jìn)行調(diào)用;
    查找未果--》向接受者所屬類的父類遞進(jìn)查找;
    查找未果--》進(jìn)行消息轉(zhuǎn)發(fā)
    在匹配成功后,objc_msgSend會(huì)將匹配結(jié)果緩存在“快速映射表”中,再次調(diào)用時(shí)速度會(huì)大大加快

  • 用方法調(diào)配技術(shù)互換兩個(gè)方法的實(shí)現(xiàn):
    Method method1 = class_getInstanceMethod([self class], @selector(testMethodWithInfo:));
    Method method2 = class_getInstanceMethod([self class], @selector(test));
    method_exchangeImplementations(method1, method2);
    系統(tǒng)在調(diào)用method1時(shí)就會(huì)執(zhí)行method2,調(diào)用method2時(shí)調(diào)用method1.
    通過(guò)這一方案,可以為那些“完全不知道其具體實(shí)現(xiàn)的”黑盒方法添加日志記錄功能,非常有助于程序調(diào)試,單一般也只在調(diào)試程序時(shí)用到這一方案。

  • Apple宣稱其保留所有“兩字母前綴”的權(quán)利,所以為了避免出現(xiàn)“重名符號(hào)錯(cuò)誤”,在應(yīng)用程序中聲明類名時(shí)都應(yīng)該添加“三字母前綴”,如所在公司是阿里,想要自定義一個(gè)PayView類,則應(yīng)給其命名為ALiPayView。

  • 給私有方法的名稱加上前綴,例如p_(p代表private),使其與公共方法區(qū)分開(kāi);
    不要單用一個(gè)下劃線做前綴,這種方式是蘋果公司預(yù)留方式,容易產(chǎn)生沖突;

  • 淺拷貝之后的內(nèi)容與原始內(nèi)容均指向相同對(duì)象,而深拷貝之后的內(nèi)容所指的對(duì)象是原始內(nèi)容中相關(guān)對(duì)象的一份拷貝。即淺拷貝后改變拷貝對(duì)象的值,原始內(nèi)容也會(huì)隨之改變,但是深拷貝后改變拷貝對(duì)象的值,原始內(nèi)容不會(huì)發(fā)生改變。

  • 無(wú)論當(dāng)前實(shí)例是否可變,若需獲取其可變版本的拷貝,則使用mutableCopy,若需獲取其不可變版本的拷貝,則使用copy
    對(duì)于不可變的NSArray與可變的NSMutableArray來(lái)說(shuō),下列關(guān)系總成立:
    [NSMutableArray copy] =>NSArray
    [NSArray mutableCopy] =>NSMutableArray

  • 當(dāng)類的實(shí)現(xiàn)中方法過(guò)多,導(dǎo)致代碼冗余時(shí),可以通過(guò)“分類”的方法,將類中的方法按需分類,劃入幾個(gè)分區(qū)中,便于管理與維護(hù)。

  • 在編寫程序庫(kù)用作分享時(shí),如果有一些方法,它們不是公共API的一部分,然而卻很適合在程序庫(kù)之內(nèi)使用,這時(shí)我們應(yīng)該創(chuàng)建private分類,當(dāng)程序庫(kù)的某些地方需要用到這些方法時(shí),就導(dǎo)入此分類頭文件,并且設(shè)置頭文件不隨程序庫(kù)一并公開(kāi)即可。

  • 向第三方類(公共API,如NSString)添加分類時(shí):
    1、總應(yīng)給其名稱加上你專用的前綴,避免類名沖突;
    2、總應(yīng)給其中的方法名加上你專用的前綴,避免方法重寫覆蓋;
    注:如果在項(xiàng)目中給第三方類創(chuàng)建了多個(gè)分類,并且都聲明并實(shí)現(xiàn)了方法名相同的方法,則在程序中調(diào)用該方法時(shí),實(shí)現(xiàn)的是最后一個(gè)創(chuàng)建的分類中的方法。
    3、在分類中不要定義屬性

  • 保留環(huán)(循環(huán)引用):當(dāng)兩個(gè)或兩個(gè)以上對(duì)象呈環(huán)狀相互引用時(shí),因?yàn)檠h(huán)中的對(duì)象其引用計(jì)數(shù)不會(huì)降為0,所以內(nèi)存無(wú)法釋放,會(huì)導(dǎo)致內(nèi)存泄漏情況。

  • 可用下列修飾符來(lái)改變局部變量與實(shí)例變量的語(yǔ)義:
    __strong: 默認(rèn)語(yǔ)義,保留此值。
    __unsafe_unretained: 不保留此值,這么做可能不安全,因?yàn)榈鹊皆俅问褂米兞繒r(shí),其對(duì)象可能已經(jīng)被收回了。
    __weak: 不保留此值,但是變量可以安全使用,因?yàn)槿绻到y(tǒng)把這個(gè)對(duì)象回收了,那么變量也會(huì)自動(dòng)清空。
    __autoreleasing: 把對(duì)象“按引用傳遞”給方法時(shí),使用這個(gè)特殊的修飾符,此值在方法返回時(shí)自動(dòng)釋放。

所以在程序調(diào)用block語(yǔ)法時(shí),因?yàn)閎lock會(huì)自動(dòng)保留其所捕獲的全部對(duì)象,而這其中如果有某個(gè)對(duì)象又保留了block塊本身(如self),就可能會(huì)導(dǎo)致循環(huán)引用(保留環(huán))。而使用__weak來(lái)修飾self時(shí),weakSelf的引用計(jì)數(shù)會(huì)自動(dòng)清空,就不會(huì)導(dǎo)致保留環(huán)的產(chǎn)生。

  • ARC只負(fù)責(zé)管理oc對(duì)象的內(nèi)存,要注意的是:CoreFoundation對(duì)象不歸ARC管理,開(kāi)發(fā)者必須適時(shí)調(diào)用CFRetain/CFRelease。

  • 給對(duì)象配置過(guò)的觀測(cè)行為都應(yīng)該在dealloc中注銷,ARC會(huì)自動(dòng)執(zhí)行父類的dealloc方法,所以在ARC下實(shí)現(xiàn)dealloc方法時(shí),不需要手動(dòng)調(diào)用[super dealloc]。

  • 系統(tǒng)在回收對(duì)象時(shí),可以不將其真的回收,而是把它轉(zhuǎn)化為僵尸對(duì)象。通過(guò)環(huán)境變量NSZombieEnable可開(kāi)啟此功能。

  • 系統(tǒng)會(huì)修改對(duì)象的isa指針,令其指向特殊的僵尸類,從而使對(duì)象變?yōu)榻┦瑢?duì)象。僵尸類能夠響應(yīng)所有的選擇器,響應(yīng)方式為:打印一條包含消息內(nèi)容及其接收者的消息,然后終止應(yīng)用程序。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,622評(píng)論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,716評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 178,746評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 63,991評(píng)論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,706評(píng)論 6 413
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 56,036評(píng)論 1 329
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,029評(píng)論 3 450
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 43,203評(píng)論 0 290
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,725評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,451評(píng)論 3 361
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,677評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,161評(píng)論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,857評(píng)論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 35,266評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 36,606評(píng)論 1 295
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,407評(píng)論 3 400
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,643評(píng)論 2 380

推薦閱讀更多精彩內(nèi)容