Cocoa深入淺出:動態(tài)創(chuàng)建類
在前文《深入淺出Cocoa之類與對象》一文中,我已經(jīng)詳細介紹了ObjC中的 Class 與 Object 的概念,今天我們來如何在運行時動態(tài)創(chuàng)建類。下面這個函數(shù)就是應(yīng)用前面講到的Class,MetaClass的概念,在運行時動態(tài)創(chuàng)建一個類。這個函數(shù)來自《Inside Mac OS X-The Objective-C Programming Language》。
#import <objc/objc.h> #import <objc/runtime.h> BOOL CreateClassDefinition( const char * name, const char * superclassName) { struct objc_class * meta_class; struct objc_class * super_class; struct objc_class * new_class; struct objc_class * root_class; va_list args; // 確保父類存在 super_class = (struct objc_class *)objc_lookUpClass (superclassName); if (super_class == nil) { return NO; } // 確保要創(chuàng)建的類不存在 if (objc_lookUpClass (name) != nil) { return NO; } // 查找 root class,因為 meta class 的 isa 指向 root class 的 meta class root_class = super_class; while( root_class->super_class != nil ) { root_class = root_class->super_class; } // 為 class 及其 meta class 分配內(nèi)存 new_class = calloc( 2, sizeof(struct objc_class) ); meta_class = &new_class[1]; // 設(shè)置 class new_class->isa = meta_class; new_class->info = CLS_CLASS; meta_class->info = CLS_META; // 拷貝類名字,這里為了提高效率,讓 class 與 meta class 都指向同一個類名字符串 new_class->name = malloc (strlen (name) + 1); strcpy ((char*)new_class->name, name); meta_class->name = new_class->name; // 分配并置空 method lists,我們可以在之后使用 class_addMethods 向類中增加方法 new_class->methodLists = calloc( 1, sizeof(struct objc_method_list *) ); meta_class->methodLists = calloc( 1, sizeof(struct objc_method_list *) ); // 將類加入到繼承體系中去: // 1,設(shè)置類的 super class // 2,設(shè)置 meta class 的 super class // 3,設(shè)置 meta class 的 isa new_class->super_class = super_class; meta_class->super_class = super_class->isa; meta_class->isa = (void *)root_class->isa; // 最后,將 class 注冊到運行時系統(tǒng)中 objc_addClass( new_class ); return YES; }
如果要在代碼中使用運行時相關(guān)的函數(shù),我們需要導入 libobjc.dylib,并導入相關(guān)的頭文件(比如這里的 runtime.h)。
在前文中總結(jié)到“ObjC 為每個類的定義生成兩個 objc_class ,一個即普通的 class,另一個即 metaclass。我們可以在運行期創(chuàng)建這兩個 objc_class 數(shù)據(jù)結(jié)構(gòu),然后使用 objc_addClass 動態(tài)地創(chuàng)建新的類定義。”,這在上面的代碼中就體現(xiàn)出來了:new_class 和 meta_class 就是新類所必須的兩個 objc_class。其他的代碼就不解釋了,注釋以及代碼足以自明了。
在實際的運用中,我們使用 ObjC 運行時函數(shù)來動態(tài)創(chuàng)建類:
Class objc_allocateClassPair(Class superclass, const char *name, size_t extraBytes);
譬如:
#import <objc/objc.h> #import <objc/runtime.h> void ReportFunction(id self, SEL _cmd) { NSLog(@" >> This object is %p.", self); NSLog(@" >> Class is %@, and super is %@.", [self class], [self superclass]); Class prevClass = NULL; int count = 1; for (Class currentClass = [self class]; currentClass; ++count) { prevClass = currentClass; NSLog(@" >> Following the isa pointer %d times gives %p", count, currentClass); currentClass = object_getClass(currentClass); if (prevClass == currentClass) break; } NSLog(@" >> NSObject's class is %p", [NSObject class]); NSLog(@" >> NSObject's meta class is %p", object_getClass([NSObject class])); } int main (int argc, const char * argv[]) { @autoreleasepool { Class newClass = objc_allocateClassPair([NSString class], "NSStringSubclass", 0); class_addMethod(newClass, @selector(report), (IMP)ReportFunction, "v@:"); objc_registerClassPair(newClass); id instanceOfNewClass = [[newClass alloc] init]; [instanceOfNewClass performSelector:@selector(report)]; [instanceOfNewClass release]; } return 0; }
在上面的代碼中,我們創(chuàng)建繼承自 NSString 的子類 NSStringSubclass,然后向其中添加方法 report,并在運行時系統(tǒng)中注冊,這樣我們就可以使用這個新類了。在這里使用 performSelector 來向新類的對象發(fā)送消息,可以避免編譯警告信息(因為我們并沒有聲明該類及其可響應(yīng)的消息)。
執(zhí)行結(jié)果為:
>> This object is 0x100114710. >> Class is NSStringSubclass, and super is NSString. >> Following the isa pointer 1 times gives 0x100114410 >> Following the isa pointer 2 times gives 0x100114560 >> Following the isa pointer 3 times gives 0x7fff7e257b50 >> NSObject's class is 0x7fff7e257b78 >> NSObject's meta class is 0x7fff7e257b50
根據(jù)前文中的類關(guān)系圖,我們不難從執(zhí)行結(jié)果中分析出 NSStringSubclass 的內(nèi)部類結(jié)構(gòu):
1,對象的地址為 :0x100114710
2,class 的地址為:0x100114410
3,meta class 的地址為:0x100114560
4,meta class 的 class 地址為:0x7fff7e257b50 (也是 NSObject 的 meta class)
5,NSObject 的 meta class 的 meta class 是其自身
原文:http://www.cnblogs.com/kesalin/archive/2012/01/30/objc_create_class.html