iOS开发ios复制粘贴

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

2016-11-06  本文已影响1405人  vvkeep

一、简介

二、功能实现

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

利用关联对象,可以不用每次都调用运行时方法遍历获取属性列表,提高程序效率
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.编码解码(用于归档)

///归档

上一篇下一篇

猜你喜欢

热点阅读