簡介:
分類是一種為現(xiàn)有的類添加新方法的方式怀偷。
利用Objective-C的動態(tài)運行時分配機制,可以為現(xiàn)有的類添加新方法泻拦,這種為現(xiàn)有的類添加新方法的方式稱為分類葡盗,他可以為任何類添加新的方法漾抬,包括那些沒有源代碼的類潮饱。
分類使得無需創(chuàng)建對象類的子類就能完成同樣的工作谍婉。
我們知道更啄,所有的OC類和對象稚疹,在runtime層都是用struct表示的,category也不例外祭务,在runtime層内狗,category用結(jié)構(gòu)體category_t(在objc-runtime-new.h中可以找到此定義),它包含了
- 類的名字(name)
- 類(cls)
- category中所有給類添加的實例方法的列表(instanceMethods)
- category中所有添加的類方法的列表(classMethods)
- category實現(xiàn)的所有協(xié)議的列表(protocols)
- category中添加的所有屬性(instanceProperties)
typedef struct category_t {
const char *name;
classref_t cls;
struct method_list_t *instanceMethods;
struct method_list_t *classMethods;
struct protocol_list_t *protocols;
struct property_list_t *instanceProperties;
} category_t;
主要作用:
- 將類的實現(xiàn)分散到多個不同文件或多個不同框架中义锥。
- 創(chuàng)建對私有方法的前向引用柳沙。
- 向?qū)ο筇砑臃钦絽f(xié)議。
OC語法中缨该,可以對類的實例變量加@private/@public等關(guān)鍵字進行修飾偎行。但是對于類的方法只分+開頭的類方法和-開頭的對象方法,不能對一個類的方法加@private這樣的關(guān)鍵字進行限定,那么OC中如何定義私有方法呢蛤袒?
答案是:定義在類名.h文件中的方法/屬性一定是公開的熄云,而在類名.m中的類延展(Extension)中定義的方法/屬性都是私有的∶钫妫或者不在任何地方申明缴允,只在類.m中寫實現(xiàn)代碼的方法也是私有的。
Cocoa沒有任何真正的私有方法珍德。只要知道對象支持的某個方法的名稱练般,即使該對象所在的類的接口中沒有該方法的聲明,你也可以調(diào)用該方法锈候。不過這么做編譯器會報錯薄料,但是只要新建一個該類的類別,在類別.h文件中寫上原始類該方法的聲明泵琳,類別.m文件中什么也不寫摄职,就可以正常調(diào)用私有方法了。這就是傳說中的私有方法前向引用获列。 所以說cocoa沒有真正的私有方法谷市。
使用Category需要注意的點:
- Category的方法不一定非要在@implementation中實現(xiàn),也可以在其他位置實現(xiàn)击孩,但是當(dāng)調(diào)用Category的方法時迫悠,依據(jù)繼承樹沒有找到該方法的實現(xiàn),程序則會崩潰巩梢。
- Category理論上不能添加變量创泄,但是可以使用@dynamic 來彌補這種不足。
static const void * externVariableKey = &externVariableKey;
@implementation NSObject (Category)
@dynamic variable且改;
- (id) variable
{
return objc_getAssociatedObject(self, externVariableKey);
}
- (void)setVariable:(id) variable
{
objc_setAssociatedObject(self, externVariableKey, variable, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- category的方法沒有“完全替換掉”原來類已經(jīng)有的方法验烧,也就是說如果category和原來類都有methodA,那么category附加完成之后又跛,類的方法列表里會有兩個methodA碍拆。
- category的方法被放到了新方法列表的前面,而原來類的方法被放到了新方法列表的后面慨蓝,這也就是我們平常所說的category的方法會“覆蓋”掉原來類的同名方法感混,這是因為運行時在查找方法的時候是順著方法列表的順序查找的,它只要一找到對應(yīng)名字的方法礼烈,就會罷休弧满,殊不知后面可能還有一樣名字的方法。
// 備注:Person 類及他的分類都有一個 printName 方法此熬,由于之前的原因庭呜,直接調(diào)用 printName 方法會直接調(diào)用分類的實現(xiàn)滑进,所以我們需要在 methodList 中找到后面的實現(xiàn),然后調(diào)用對應(yīng)的方法募谎。
// 獲取 Person 類本身的 printName 方法
unsigned int methodCount;
Method *methodList = class_copyMethodList([Person class], &methodCount);
IMP lastImp = NULL;
SEL lastSel = NULL;
for (NSUInteger i = 0; i < methodCount; i++) {
Method method = methodList[i];
NSString *methodName = [NSString stringWithCString:sel_getName(method_getName(method)) encoding:NSUTF8StringEncoding];
if ([@"printName" isEqualToString:methodName]) {
lastImp = method_getImplementation(method);
lastSel = method_getName(method);
}
}
typedef void(*imp)(id, SEL);
if (lastImp != NULL) {
imp f = (imp)lastImp;
f(p, lastSel);
}
free(methodList);