runtime

作者: KnowWhy | 来源:发表于2017-05-02 00:09 被阅读0次

    运行时,就是尽可能地把决定从编译器推迟到运行期,就是尽可能地做到动态。只是在运行的时候才会去确定对象的类型和方法。 因此利用Runtime机制可以在程序运行时动态地修改类和对象中的所有属性和方法。

    Runtime是一套比较底层的C语言的API,Objective-C是运行在Runtime上的,因此在Runtime中动态添加和实现一些非常强大的功能也就不足为奇了。

    在Objective-C代码中使用Runtime, 需要引入头文件:

    #import <objc/runtime.h>
    

    一、归档、解档进行数据持久化

    比如自建模型类Student,使用NSKeyedArchiver类序列化,持久化数据,NSKeyedUnarchiver反序列化。序列化和反序列化需要遵循NSCoding协议。NSCoding协议实现举例如下:

    // Student.m文件方法实现部分
    // 注意:归档解档需要遵守<NSCoding>协议,实现以下两个方法
    /**
     归档
    
     @param aCoder 编码器
     */
    - (void)encodeWithCoder:(NSCoder *)aCoder
    {
        unsigned int count = 0;
        // 获得指向该类实例变量的指针
        Ivar *ivars = class_copyIvarList([self class], &count);
        for (int i = 0; i < count; i++) {
            Ivar ivar = ivars[i];   // 获取某个实例变量(C语言的字符串)
            const char *name = ivar_getName(ivar);
            NSString *key = [NSString stringWithUTF8String:name];
            // 编码每个实例变量,利用kVC取出每个实例变量对应的数值
            [aCoder encodeObject:[self valueForKeyPath:key] forKey:key];
        }
        free(ivars);    // 记得释放C定义的变量
    }
    
    /**
     解档
    
     @param aDecoder 解码器
     @return 实例
     */
    - (instancetype)initWithCoder:(NSCoder *)aDecoder
    {
        self = [super init];
        if (self) {
            unsigned int count = 0;
            // 获得指向该类所有属性的指针
            Ivar *ivars = class_copyIvarList([self class], &count);
            for (int i = 0; i < count; i++) {
                Ivar ivar = ivars[i];
                const char *name = ivar_getName(ivar);
                NSString *key = [NSString stringWithUTF8String:name];
                // 解码每个属性,利用kVC取出每个实例变量对应的数值
                [self setValue:[aDecoder decodeObjectForKey:key] forKeyPath:key];
            }
            free(ivars);    // 记得释放C定义的变量
        }
        return self;
    }
    

    二、Method Swizzling方法实现交换

    @implementation UIImage (Image)
    // 加载分类到内存的时候调用
    + (void)load
    {
      // 交换方法实现,方法都是定义在类里面
      // class_getMethodImplementation:获取方法实现
      // class_getInstanceMethod:获取实例方法
      // class_getClassMethod:获取类方法
      // IMP:方法实现
     
      // imageNamed
      // Class:获取哪个类方法
      // SEL:获取方法编号,根据SEL就能去对应的类找方法
     
      Method imageNameMethod = class_getClassMethod([UIImage class], @selector(imageNamed:));
     
      Method SW_imageNameMethod = class_getClassMethod([UIImage class], @selector(SW_imageNamed:));
     
      // 交换方法实现
      method_exchangeImplementations(imageNameMethod, SW_imageNameMethod);
    }
     
    // 不能在分类中重写系统方法imageNamed,因为会把系统的功能给覆盖掉,而且分类中不能调用super.
    // 既能加载图片又能打印
    + (UIImage *)SW_imageNamed:(NSString *)imageName
    {
      // 加载图片
      UIImage *image = [UIImage SW_imageNamed:imageName];
      // 2.判断功能
      if (image == nil) {
        NSLog(@"加载为空");
      }
     
      return image;
    }
    @end
    

    三、绑定实例对象传递

    // _cmd在Objective-C的方法中表示当前方法的selector,正如同self表示当前方法调用的对象实例。
    - (BOOL)fd_debugLogEnabled {
        return [objc_getAssociatedObject(self, _cmd) boolValue];
    }
    
    - (void)setFd_debugLogEnabled:(BOOL)debugLogEnabled {
        objc_setAssociatedObject(self, @selector(fd_debugLogEnabled), @(debugLogEnabled), OBJC_ASSOCIATION_RETAIN);
    }
    
    id objc_getAssociatedObject(id object, const void *key); //获取传递的实例对象
    void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);// 设置传递的实例对象
    

    相关文章

      网友评论

          本文标题:runtime

          本文链接:https://www.haomeiwen.com/subject/mumqtxtx.html