[Cocoa]深入浅出Cocoa 之动态创建类

本文介绍如何使用Objective-C运行时API动态创建类及其元类,包括objc_allocateClassPair和objc_registerClassPair等函数的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在前文 《深入浅出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 是其自身
基于Spring Boot搭建的一个多功能在线学习系统的实现细节。系统分为管理员和用户两个主要模块。管理员负责视频、文件和文章资料的管理以及系统运营维护;用户则可以进行视频播放、资料下载、参与学习论坛并享受个性化学习服务。文中重点探讨了文件下载的安全性和性能优化(如使用Resource对象避免内存溢出),积分排行榜的高效实现(采用Redis Sorted Set结构),敏感词过滤机制(利用DFA算法构建内存过滤树)以及视频播放的浏览器兼容性解决方案(通过FFmpeg调整MOOV原子位置)。此外,还提到了权限管理方面自定义动态加载器的应用,提高了系统的灵活性和易用性。 适合人群:对Spring Boot有一定了解,希望深入理解其实际应用的技术人员,尤其是从事在线教育平台开发的相关从业者。 使用场景及目标:适用于需要快速搭建稳定高效的在线学习平台的企业或团队。目标在于提供一套完整的解决方案,涵盖从资源管理到用户体验优化等多个方面,帮助开发者更好地理解和掌握Spring Boot框架的实际运用技巧。 其他说明:文中不仅提供了具体的代码示例和技术思路,还分享了许多实践经验教训,对于提高项目质量有着重要的指导意义。同时强调了安全性、性能优化等方面的重要性,确保系统能够应对大规模用户的并发访问需求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值