前言:第一次閱讀此書(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)用程序芥炭。