深入浅出Cocoa 之动态创建类

深入浅出Cocoa 之动态创建类,第1张

概述在前文《深入浅出Cocoa之类与对象》一文中,我已经详细介绍了ObjC中的 Class 与 Object 的概念,今天我们来如何在运行时动态创建类。 在前文《深入浅出Cocoa之类与对象》一文中,我已经详细介绍了ObjC中的 Class 与 Object 的概念,今天我们来如何在运行时动态创建类。下面这个函数就是应用前面讲到的Class,MetaClass的概念,在运行时动态创建一个类。这个函数来

在前文《深入浅出Cocoa之类与对象》一文中,我已经详细介绍了ObjC中的 Class 与 Object 的概念,今天我们来如何在运行时动态创建类。

在前文《深入浅出Cocoa之类与对象》一文中,我已经详细介绍了ObjC中的 Class 与 Object 的概念,今天我们来如何在运行时动态创建类。下面这个函数就是应用前面讲到的Class,MetaClass的概念,在运行时动态创建一个类。这个函数来自《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;     }          // 确保要创建的类不存在     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 分配内存     new_class = calloc( 2,sizeof(struct objc_class) );     Meta_class = &new_class[1];          // 设置 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,设置类的 super class     // 2,设置 Meta class 的 super class     // 3,设置 Meta class 的 isa     new_class->super_class = super_class;     Meta_class->super_class = super_class->isa;     Meta_class->isa = (voID *)root_class->isa;          // 最后,将 class 注册到运行时系统中     objc_addClass( new_class );          return YES; }     如果要在代码中使用运行时相关的函数,我们需要导入 libobjc.dylib,并导入相关的头文件(比如这里的 runtime.h)。   在前文中总结到“ObjC 为每个类的定义生成两个 objc_class ,一个即普通的 class,另一个即 Metaclass。我们可以在运行期创建这两个 objc_class 数据结构,然后使用 objc_addClass 动态地创建新的类定义。”,这在上面的代码中就体现出来了:new_class 和 Meta_class 就是新类所必须的两个 objc_class。其他的代码就不解释了,注释以及代码足以自明了。   在实际的运用中,我们使用 ObjC 运行时函数来动态创建类: 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; }   在上面的代码中,我们创建继承自 Nsstring 的子类 NsstringSubclass,然后向其中添加方法 report,并在运行时系统中注册,这样我们就可以使用这个新类了。在这里使用 performSelector 来向新类的对象发送消息,可以避免编译警告信息(因为我们并没有声明该类及其可响应的消息)。   执行结果为:    >> 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     根据前文中的类关系图,我们不难从执行结果中分析出 NsstringSubclass 的内部类结构: 1,对象的地址为 :0x100114710 2,class 的地址为:0x100114410 3,Meta class 的地址为:0x100114560 4,Meta class 的 class 地址为:0x7fff7e257b50 (也是 NSObject 的 Meta class) 5,NSObject 的 Meta class 的 Meta class 是其自身     转自: 飘飘白云 @H_404_295@ 总结

以上是内存溢出为你收集整理的深入浅出Cocoa 之动态创建类全部内容,希望文章能够帮你解决深入浅出Cocoa 之动态创建类所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1059817.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-25
下一篇 2022-05-25

发表评论

登录后才能评论

评论列表(0条)

保存