美文网首页iOS开发ios复制粘贴
Runtime - 运行时,项目中的实战用法看我就够了

Runtime - 运行时,项目中的实战用法看我就够了

作者: vvkeep | 来源:发表于2016-11-06 14:21 被阅读1405次

    一、简介

    • 运行时是一种面向对象的编程语言环境,类似于java的虚拟机
    • OC最主要的特点就是在程序运行的时候,以发送消息的方式调用方法
    • 运行时是OC的核心、底层,OC就是基于运行时的
    • 日常工作中,主要应用场景是关联对象、可以给分类动态的添加属性;动态的获取类的属性,用于字典转模型
    • 交叉方法,在无法修改系统的或者第三方框架的方式时,利用交叉方法,先执行自己的方法,在交换执行三方框架的方法,

    二、功能实现

    • Runtime提供获取类信息的方法如下:


      获取类的方法.png

    通过以上的函数,可以获取类的属性、协议、成员变量、和方法

    • 关联对象

    利用关联对象,可以不用每次都调用运行时方法遍历获取属性列表,提高程序效率
    void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
    此方法用于动态创建属性,记录属性数组

    id objc_getAssociatedObject(id object, const void *key)
    调用运行性的方法,判断对象属性是否已经获取,如果获取直接返回

    1.动态获取类的属性列表

    /**
     获取类的属性列表数组
    
     @return 类的属性列表数组
     */
    + (NSArray *)yw_objPropertyArr {
        
        //从关联对象 中获取对象属性,如果有,直接返回
        /**
         参数:
         1 对象 self
         2 动态属性的key
         返回值 id 动态添加的 属性值
         */
        NSArray *ptyList = objc_getAssociatedObject(self, KPropertyListKey);
        if (ptyList) {
            return ptyList;
        }
       
        /**
         获取属性列表
         1.要获取的类
         2.类属性的个数指针
         返回值:所有的属性数组 C语言中,数组的名字,就是指向第一个元素的地址 
        在OC 中使用C的时候晕倒 retain/create/copy 等 需要release
         */
        unsigned int count = 0;
        //C语言数组 需要 * 符号
        objc_property_t *proArr = class_copyPropertyList([self class], &count);
        NSLog(@"属性的数量%d",count);
        //创建数组
        NSMutableArray *MArr = [NSMutableArray array];
        //遍历所有的属性
        for (unsigned int i = 0; i < count; i++) {
            //从数组中取得属性
            // C语言结构体指针,不要 *
            objc_property_t pty = proArr[i];
            //从pty 中获取属性的名称
            const char *cName = property_getName(pty);
            NSString *name = [NSString stringWithCString:cName encoding:NSUTF8StringEncoding];
            //添加到数组
            [MArr addObject:name];
        }
        //释放数组
        free(proArr);
        
        //到此为止,对象的属性数组获取完毕,利用关联对象,动态的添加属性
        /**
         1.对象 self
         2.动态添加属性的key,获取值的时候使用
         3.动态添加属性值
         4.对象的引用关系
         */
        objc_setAssociatedObject(self, KPropertyListKey, MArr.copy, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
        return MArr.copy;
    }
    

    2.字典转模型

    /**
     给定一个字典,创建self类 对用的对象
     @param dic 字典
     @return 对象
     */
    + (instancetype)yw_objWithDic:(NSDictionary *)dic {
        //实例化对象
        id object = [[self alloc] init];
        //使用字典使用对象信息
        //获取self 是属性列表
        NSArray *proArr = [self yw_objPropertyArr];
        //遍历字典的方法
        [dic enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
            NSLog(@"key %@, value %@",key,obj);
            //判断key 是否在proArr中
            if ([proArr containsObject:key]) {
                //属性存在KVC 赋值
                [object setValue:obj forKey:key];
            }
        }];
        return object;
    }
    

    3.交叉方法

    举个例子,在imageView setImage的时候,会根据imageView的大小对图片进行缩放,这样的性能不是好,特别是在表格滚动的时候,这时,就可以使用交叉的黑魔法,通过上下文绘制的方法,解决这一问题:

    /**
     类被加载到运行时,就会执行
     */
    + (void)load {
        Method originalMethod = class_getInstanceMethod([self class], @selector(setImage:));
        Method swizzledMethod = class_getInstanceMethod([self class], @selector(yw_setImage:));
        method_exchangeImplementations(originalMethod, swizzledMethod);
    }
    - (void)yw_setImage:(UIImage *)image {
        NSLog(@"%s",__func__);
        //根据imageView大小,重新调整 image 大小
        UIGraphicsBeginImageContextWithOptions(self.bounds.size, YES, 0);
        //绘制图像
        [image drawInRect:self.bounds];
        //取得结果
        UIImage *result = UIGraphicsGetImageFromCurrentImageContext();
        //关闭上下文
        UIGraphicsEndImageContext();
        //此处 setImage 和 yw_setImage方法已经被交换
        //调用系统原生的setImage方法
        [self yw_setImage:result];
    } ```
    
    ####4.编码解码(用于归档)
    

    ///归档

    • (void)encode:(NSCoder *)aCoder{
      unsigned int outCount = 0;
      Ivar *ivars = class_copyIvarList([self class], &outCount);
      for (unsigned int i = 0; i < outCount; i++)
      {
      Ivar ivar = ivars[i];
      NSString *key = [NSString stringWithUTF8String:ivar_getName(ivar)];
      if ([self.ignoredIvarNames containsObject:key])
      {
      continue;
      }
      id value = [self valueForKey:key];
      [aCoder encodeObject:value forKey:key];
      }
      free(ivars);
      }
      ///解档

    • (void)decode:(NSCoder *)aDecoder{
      unsigned int outCount = 0;
      Ivar *ivars = class_copyIvarList([self class], &outCount);
      for (unsigned int i = 0; i < outCount; i++)
      {
      Ivar ivar = ivars[i];
      NSString *key = [NSString stringWithUTF8String:ivar_getName(ivar)];
      if ([self.ignoredIvarNames containsObject:key])
      {
      continue;
      }
      id value = [aDecoder decodeObjectForKey:key];
      [self setValue:value forKey:key];
      }
      free(ivars);
      }
      ///忽略数组

    • (void)setIgnoredIvarNames:(NSArray *)ignoredIvarNames{
      objc_setAssociatedObject(self,
      @selector(ignoredIvarNames),
      ignoredIvarNames,
      OBJC_ASSOCIATION_RETAIN_NONATOMIC);
      }

    • (NSArray )ignoredIvarNames{
      return objc_getAssociatedObject(self, _cmd);
      }```
      githubDemo:
      runtime01*

    相关文章

      网友评论

      • aecbc2bdc4cb:”成员变量、和方法”这里不应该加“、”呀:stuck_out_tongue_winking_eye:
      • iOS_Ken:可以动态获取对象的类型~~?
      • iOS_Ken:其实运行时也相当于java的反射!
        theonlin:@hard巍 为什么这么说?
        vvkeep:@c7a707cf93bf 反射只是运行时的一部分
      • 圣罗迦奈:蛤膜拜大神
      • iYeso:留个联系方式吧 请教请教:smile:
      • xiAo__Ju:喵喵,问个问题,在Swift中用这个object_setIvar我想设置值,但是没用啊!
        xiAo__Ju:@xiAo__Ju http://stackoverflow.com/questions/25718294/how-do-i-use-objective-c-runtimes-object-getivar-object-setivar-in-swift,这里说Swift中,没有这个东东了
      • 念奴娇如画:我知道了,我写到 uIImage的分类里去了
      • 念奴娇如画: UIGraphicsBeginImageContextWithOptions(self.bounds.size, YES, 0);
        //绘制图像
        [image drawInRect:self.bounds];
        这个bounds报错,什么原因,大神
      • hauibojek:高产似母🐷😄
        vvkeep:@HUswIft :smile:
        hauibojek: @iOS巍 好说好说 哈哈
        vvkeep:@HUswIft 也不点个赞 :cry:
      • xiAo__Ju:最近很高产啊!
        xiAo__Ju:@iOS巍 打算等哪天慢慢吸收完!
        vvkeep:@xiAo__Ju 也不点个赞 :cry:

      本文标题:Runtime - 运行时,项目中的实战用法看我就够了

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