分類
- 一個指向分類的的結(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方法聲明(嚴謹來說,通過正常的方法是不能聲明屬性的,如果聲明了屬性無法自動合成setter、getter方法,但是可以通過 runtime來增加屬性)
- 類擴展不僅可以增加方法,還可以增加屬性、成員變量,只是該成員變量默認是@private類型的(作用范圍只能在自身類,而不是子類或其他地方)
- 類擴展不能像類別那樣擁有獨立的實現(xiàn)部分(@implementation部分),也就是說,類擴展所聲明的方法必須依托對應(yīng)類的實現(xiàn)部分來實現(xiàn)