分類
- 一個指向分類的的結(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)