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

前言:第一次閱讀此書(shū)大概是一年半之前琉雳,在網(wǎng)上找到電子版益缠,也就臨時(shí)看了一晚上蒂胞,之后就沒(méi)有再涉獵税手。八月份決定抽出半個(gè)月左右的時(shí)間認(rèn)真閱讀下這本書(shū)(但是由于這兩周一直在準(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è)下劃線做前綴悦屏,這種方式是蘋(píng)果公司預(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ù)卧波。

  • 在編寫(xiě)程序庫(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ě)覆蓋炭臭;
    注:如果在項(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)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市恃慧,隨后出現(xiàn)的幾起案子园蝠,更是在濱河造成了極大的恐慌,老刑警劉巖痢士,帶你破解...
    沈念sama閱讀 221,695評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件彪薛,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡怠蹂,警方通過(guò)查閱死者的電腦和手機(jī)善延,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)城侧,“玉大人易遣,你說(shuō)我怎么就攤上這事∠佑樱” “怎么了豆茫?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,130評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)屋摇。 經(jīng)常有香客問(wèn)我揩魂,道長(zhǎng),這世上最難降的妖魔是什么炮温? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,648評(píng)論 1 297
  • 正文 為了忘掉前任火脉,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘倦挂。我一直安慰自己畸颅,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,655評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布妒峦。 她就那樣靜靜地躺著重斑,像睡著了一般兵睛。 火紅的嫁衣襯著肌膚如雪肯骇。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,268評(píng)論 1 309
  • 那天祖很,我揣著相機(jī)與錄音笛丙,去河邊找鬼。 笑死假颇,一個(gè)胖子當(dāng)著我的面吹牛胚鸯,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播笨鸡,決...
    沈念sama閱讀 40,835評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼姜钳,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了形耗?” 一聲冷哼從身側(cè)響起哥桥,我...
    開(kāi)封第一講書(shū)人閱讀 39,740評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎激涤,沒(méi)想到半個(gè)月后拟糕,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,286評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡倦踢,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,375評(píng)論 3 340
  • 正文 我和宋清朗相戀三年送滞,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片辱挥。...
    茶點(diǎn)故事閱讀 40,505評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡犁嗅,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出晤碘,到底是詐尸還是另有隱情褂微,我是刑警寧澤,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布哼蛆,位于F島的核電站蕊梧,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏腮介。R本人自食惡果不足惜肥矢,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,873評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧甘改,春花似錦旅东、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,357評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至忘嫉,卻和暖如春荤牍,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背庆冕。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,466評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工康吵, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人访递。 一個(gè)月前我還...
    沈念sama閱讀 48,921評(píng)論 3 376
  • 正文 我出身青樓晦嵌,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親拷姿。 傳聞我的和親對(duì)象是個(gè)殘疾皇子惭载,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,515評(píng)論 2 359

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