關(guān)于Category 的用法以及底層原理的問題,iOS Category的本質(zhì)(一)這篇文章已經(jīng)講的很清楚了,今天主要講講Extension的用法以及和Category的區(qū)別。
類擴(kuò)展就像匿名的分類一樣,除了一樣不同的是,類擴(kuò)展聲明必須在@implementation在實現(xiàn)。
新建一個Cat類,代碼如下:
#import "Cat.h"
@interface Cat ()
/*** 名字 ****/
@property(nonatomic,copy) NSString *name;
- (void)eatFood;
@end
@implementation Cat
- (void)eatFood{
}
@end
聲明的方法必須要實現(xiàn),不然編譯器會提出警告。
從上面看出,分類和類擴(kuò)展的相似之處是:都可以為類添加一個額外的方法;
不同之處在于:要添加額外方法,分類必須在第一個@interface中聲明方法,并且在@implementation中提供實現(xiàn),不然運行時出錯。而類擴(kuò)展,你添加的方法是一個required API,如果不去實現(xiàn),編譯器會警告,而且這個方法的聲明可以不在第一個@interface中去聲明。
區(qū)別:
分類:是不可以聲明實例變量,通常是公開的,文件名通常為:”主類類名+分類類名.h”
擴(kuò)展:是可以聲明實例變量,是私有的,文件名通常為:”主類類名_擴(kuò)展標(biāo)識.h”,注意擴(kuò)展沒有名的。
區(qū)別分類與擴(kuò)展
1.都可以在主類中聲明使用
2.通常來講由于分類不能創(chuàng)建實例變化,本質(zhì)上與主類有區(qū)別,所以不建議寫在主類中。
3.擴(kuò)展與主類緊密聯(lián)系在一起,可以創(chuàng)建實例變量,所以通常來講會把擴(kuò)展和主類創(chuàng)建在一起。
4.擴(kuò)展(Extension)中的方法和屬性在編譯時就已經(jīng)存儲在類信息中了。類別(Category)中類的方法在運行時才存儲在類信息中。
編譯時:
編譯時顧名思義就是正在編譯的時候.那啥叫編譯呢?就是編譯器幫你把源代碼翻譯成機(jī)器能識別的代碼.(當(dāng)然只是一般意義上這么說,實際上可能只是翻譯成某個中間狀態(tài)的語言.比如Java只有JVM識別的字節(jié)碼,C#中只有CLR能識別的MSIL.另外還有啥鏈接器.匯編器.為了了便于理解我們可以統(tǒng)稱為編譯器)
那編譯時就是簡單的作一些翻譯工作,比如檢查老兄你有沒有粗心寫錯啥關(guān)鍵字了啊.有啥詞法分析,語法分析之類的過程.就像個老師檢查學(xué)生的作文中有沒有錯別字和病句一樣.如果發(fā)現(xiàn)啥錯誤編譯器就告訴你.如果你用微軟的VS的話,點下build.那就開始編譯,如果下面有errors或者warning信息,那都是編譯器檢查出來的.所謂這時的錯誤就叫編譯時錯誤,這個過程中做的啥類型檢查也就叫編譯時類型檢查,或靜態(tài)類型檢查(所謂靜態(tài)嘛就是沒把真把代碼放內(nèi)存中運行起來,而只是把代碼當(dāng)作文本來掃描下).所以有時一些人說編譯時還分配內(nèi)存啥的肯定是錯誤的說法.
運行時:
所謂運行時就是代碼跑起來了.被裝載到內(nèi)存中去了.(你的代碼保存在磁盤上沒裝入內(nèi)存之前是個死家伙.只有跑到內(nèi)存中才變成活的).而運行時類型檢查就與前面講的編譯時類型檢查(或者靜態(tài)類型檢查)不一樣.不是簡單的掃描代碼.而是在內(nèi)存中做些操作,做些判斷.
相關(guān)問題整理
1、為什么OC不能sizeof一個對象的大?。亢皖惤Y(jié)構(gòu)相近的結(jié)構(gòu)體卻可以
因為oc的動態(tài)繼承編譯機(jī)制,動態(tài)繼承機(jī)制,就是說在編譯的時候不能確定父類的大小,只有在運行時才能確定父類大小,
sizeo是在棧中操作的,編譯的時候就會計算出來sizeof的值
而棧中不知道對象的父類大小,所以不能使使用sizeof計算出對象的大小
2、為什么OC不能將對象聲明到靜態(tài)空間,如棧中,和類相近的結(jié)構(gòu)體卻可以
棧是在編譯完成后產(chǎn)生的,編譯的結(jié)果是二進(jìn)制機(jī)器文件,即匯編棧已經(jīng)產(chǎn)生,所以棧內(nèi)不能放類的對象,因為產(chǎn)生棧的時候不知道父類的大小
棧是由匯編代碼指令描述的
為什么結(jié)構(gòu)體可以直接寄計算sizeof,也能直接聲明到靜態(tài)空間呢?
為什么結(jié)構(gòu)體不管定義在棧中或堆中都能直接sizeof?因為定義到堆中的時候已經(jīng)知道了結(jié)構(gòu)體的大小,因為結(jié)構(gòu)體對于調(diào)用它的代碼產(chǎn)生的棧而言是暴露的。
是因為結(jié)構(gòu)體定義的頭文件直接把結(jié)構(gòu)體的屬性暴露給了棧,所以可以直接聲明到靜態(tài)空間
總之一句話,父類的詳細(xì)情況對于調(diào)用其子類的棧來說是封閉的,而結(jié)構(gòu)體相對于調(diào)用它的棧來說是暴露的。
OC中的動態(tài)繼承編譯機(jī)制是在編譯的時候不把父類詳細(xì)情況暴露給調(diào)用子類的棧,而是在運行的時候才把父類的詳細(xì)情況暴露給調(diào)用子類的棧