分類(Category)和類擴展(Class Extension)

分類

  • 一個指向分類的的結(jié)構(gòu)體指針
  • 原則上只能增加方法批狐,不能增加成員(實例)變量

Category源碼

typedef struct objc_category *Category;

struct objc_category {
    char *category_name                                      OBJC2_UNAVAILABLE; // 分類名字
    char *class_name                                         OBJC2_UNAVAILABLE; // 分類所屬類的類名
    struct objc_method_list *instance_methods                OBJC2_UNAVAILABLE; // 實例方法列表
    struct objc_method_list *class_methods                   OBJC2_UNAVAILABLE; // 類方法列表
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE; // 分類實現(xiàn)的協(xié)議列表
} 

在這個結(jié)構(gòu)體里,我們發(fā)現(xiàn)里面根本就沒有屬性列表寡痰!沒有屬性列表朋鞍!沒有屬性列表!

注意
  • 原則上講它只能添加方法, 不能添加屬性和成員變量朝巫。因為分類的結(jié)構(gòu)體指針中鸿摇,沒有屬性列表,只有方法列表劈猿。實際上可以通過其它方式添加屬性拙吉。
  • 分類中的可以寫@property, 但不會生成setter/getter方法, 也不會生成實現(xiàn)以及私有的成員變量(編譯時會報警告);
  • 可以在分類中訪問原有類中.h中的屬性;
  • 如果分類中有和原有類同名的方法, 會優(yōu)先調(diào)用分類中的方法, 就是說會忽略原有類的方法。所以同名方法調(diào)用的優(yōu)先級為分類 > 本類 > 父類揪荣。因此在開發(fā)中盡量不要覆蓋原有類;
  • 如果多個分類中都有和原有類中同名的方法, 那么調(diào)用該方法的時候執(zhí)行誰由編譯器決定筷黔;編譯器會執(zhí)行最后一個參與編譯的分類中的方法。

分類不能添加屬性的實質(zhì)原因

在一個類中用@property聲明一個屬性時仗颈,編譯器會自動的幫我們生成一個下劃線開頭的成員變量和對應(yīng)的setter佛舱、getter方法椎例。但是在分類的結(jié)構(gòu)體指針中,我們發(fā)現(xiàn)里面并沒有屬性列表请祖。所以在分類中用@property聲明屬性時订歪,既無法生成下劃線開頭的成員變量也無法生成setter、getter方法肆捕。

代碼

@interface UIViewController (Category)
@property (nonatomic, copy) NSString *nameWithSetterGetter;
@property (nonatomic, copy) NSString *nameWithoutSetterGetter;
- (void)programCategoryMethod;
@end

可以正常編譯刷晋,但是會用警告。

Property 'nameWithSetterGetter' requires method 'nameWithSetterGetter' to be defined - use @dynamic or provide a method implementation in this category
Property 'nameWithSetterGetter' requires method 'setNameWithSetterGetter:' to be defined - use @dynamic or provide a method implementation in this category
Property 'nameWithoutSetterGetter' requires method 'nameWithoutSetterGetter' to be defined - use @dynamic or provide a method implementation in this category
Property 'nameWithoutSetterGetter' requires method 'setNameWithoutSetterGetter:' to be defined - use @dynamic or provide a method implementation in this category

編譯器檢測出沒有相應(yīng)的setter慎陵、getter方法眼虱。這個時候,只要不用到對應(yīng)的setter席纽、getter方法就不會報錯捏悬。而且我們發(fā)現(xiàn)不能使用相應(yīng)的成員變量(__nameWithSetterGetter_nameWithoutSetterGetter)。

self.nameWithoutSetterGetter = @"無setter/getter";    // 調(diào)用setter润梯,編譯成功邮破,運行報錯為: -[ViewController setNameWithSetterGetter:]: unrecognized selector sent to instance '
NSLog(@"%@",self.nameWithoutSetterGetter);           // 調(diào)用getter,編譯成功仆救,運行報錯為:-[ViewController setNameWithSetterGetter:]: unrecognized selector sent to instance'
NSLog(@"%@",_nameWithoutSetterGetter);        //這是調(diào)用_成員變量,錯誤提示為:(Use of undeclared identifier '_nameWithoutSetterGetter')

既然報錯的根本原因是使用了系統(tǒng)沒有生成的setter/getter方法,可不可以在手動添加setter/getter來避免崩潰矫渔,完成調(diào)用呢彤蔽? 其實是可以的。由于OC是動態(tài)語言庙洼,方法真正的實現(xiàn)是通過runtime完成的顿痪,雖然系統(tǒng)不給我們生成setter/getter,但我們可以通過runtime手動添加setter/getter方法油够。那具體怎么實現(xiàn)呢蚁袭?

#import <objc/runtime.h>

static char nameWithSetterGetterKey;

@implementation UIViewController (Category)

- (void)setNameWithSetterGetter:(NSString *)nameWithSetterGetter{
    objc_setAssociatedObject(self, &nameWithSetterGetterKey, nameWithSetterGetter, OBJC_ASSOCIATION_COPY);
}

- (NSString *)nameWithSetterGetter{
    return objc_getAssociatedObject(self, &nameWithSetterGetterKey);
}
@end

但是注意,以上代碼僅僅是手動實現(xiàn)了setter/getter方法石咬,但調(diào)用_****成員變量**依然報錯揩悄。


類擴展(Class Extension)

Extension是Category的一個特例。類擴展與分類相比只少了分類的名稱鬼悠,所以稱之為“匿名分類”删性。

  • 類擴展格式

      @interface XXX ()
      // 私有屬性
      // 私有方法(如果不實現(xiàn),編譯時會報警,Method definition for 'XXX' not found)
      @end
    
  • 作用

1.為一個類添加額外的原來沒有變量焕窝,方法和屬性

2.一般的類擴展寫到.m文件中

3.一般的私有屬性寫到.m文件中的類擴展中


分類與類擴展的區(qū)別

  • 分類有名字蹬挺,類擴展沒有名字
  • 分類只能擴充方法,不能擴充成員變量,如果在分類中聲明了一個屬性,分類只會生成這個屬性的get、set方法聲明(嚴(yán)謹(jǐn)來說它掂,通過正常的方法是不能聲明屬性的巴帮,如果聲明了屬性無法自動合成setter、getter方法,但是可以通過 runtime來增加屬性)
  • 類擴展不僅可以增加方法榕茧,還可以增加屬性垃沦、成員變量,只是該成員變量默認(rèn)是@private類型的(作用范圍只能在自身類雪猪,而不是子類或其他地方)
  • 類擴展不能像類別那樣擁有獨立的實現(xiàn)部分(@implementation部分)栏尚,也就是說,類擴展所聲明的方法必須依托對應(yīng)類的實現(xiàn)部分來實現(xiàn)
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末只恨,一起剝皮案震驚了整個濱河市译仗,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌官觅,老刑警劉巖纵菌,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異休涤,居然都是意外死亡咱圆,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進(jìn)店門功氨,熙熙樓的掌柜王于貴愁眉苦臉地迎上來序苏,“玉大人,你說我怎么就攤上這事捷凄〕老辏” “怎么了?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵跺涤,是天一觀的道長匈睁。 經(jīng)常有香客問我,道長桶错,這世上最難降的妖魔是什么航唆? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮院刁,結(jié)果婚禮上糯钙,老公的妹妹穿的比我還像新娘。我一直安慰自己黎比,他們只是感情好超营,可當(dāng)我...
    茶點故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著阅虫,像睡著了一般演闭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上颓帝,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天米碰,我揣著相機與錄音窝革,去河邊找鬼。 笑死吕座,一個胖子當(dāng)著我的面吹牛虐译,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播吴趴,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼漆诽,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了锣枝?” 一聲冷哼從身側(cè)響起厢拭,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎撇叁,沒想到半個月后供鸠,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡陨闹,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年楞捂,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片趋厉。...
    茶點故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡寨闹,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出君账,到底是詐尸還是另有隱情鼻忠,我是刑警寧澤,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布杈绸,位于F島的核電站,受9級特大地震影響矮瘟,放射性物質(zhì)發(fā)生泄漏瞳脓。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一澈侠、第九天 我趴在偏房一處隱蔽的房頂上張望劫侧。 院中可真熱鬧,春花似錦哨啃、人聲如沸烧栋。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽审姓。三九已至,卻和暖如春祝峻,著一層夾襖步出監(jiān)牢的瞬間魔吐,已是汗流浹背扎筒。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留酬姆,地道東北人嗜桌。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像辞色,于是被迫代替她去往敵國和親骨宠。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,619評論 2 354

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