Runtime 在项目中的用处.

2017-11-18  本文已影响0人  Damoness

1. 全局替换系统的方法为自己的方法

使用UIImage imageNamed 加载图片时,由于需要手动输入名称,这个很容易导致加载的图片对象为空, 这个问题很容易忽视. 这时如果替换掉系统的方法, 并判断是否为空,并打印出来 这就很直观.

@implementation UIImage (image)


//分类被加入运行时的时候就调用,在main方法之前.
+(void)load{
    
    // 1.获取系统方法
    Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));
    
    // 2.自己实现的方法
    Method ln_imageNamedMethod = class_getClassMethod(self, @selector(ln_imageNamed:));
    
    
    //3.交换方法
    method_exchangeImplementations(imageNamedMethod, ln_imageNamedMethod);
}

+(UIImage *)ln_imageNamed:(NSString *)name{
    
    //这里是关键,因为ln_imageNamed已经 和 imageNamed交换,这里实际调用的是imageNamed,并不会导致循环调用.
    UIImage *image = [UIImage ln_imageNamed:name];
    if (image) {
        NSLog(@"runtime添加额外功能--加载成功");
    }else{
        NSLog(@"runtime添加额外的功能--加载失败");
    }
    
    
    return image;
}


@end

//使用
[UIImage imageNamed:@"imageName"];

2. 给系统的类增加属性.

系统的类不能满足自己的需求,需要增加额外的属性.其实本质就是给这个类添加关联,并不是直接把这个值的内存空间添加到类存空间。


@interface NSObject (Property)


@property NSString *name;


@end


@implementation NSObject (Property)

- (void)setName:(NSString *)name{
    
    //将属性的值和传递进的name关联
    objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    
}

-(NSString *)name{
    
    //获取关联属性的值
    return objc_getAssociatedObject(self, @"name");
}

@end

3. 自己实现字典转属性

@interface NSObject (DictToModel)
+(instancetype)modelWithDict:(NSDictionary *)dict;
@end

@implementation NSObject (DictToModel)


+(instancetype)modelWithDict:(NSDictionary *)dict{
    
    //1.创建对象
    id objc = [[self alloc]init];
    
    // 2.利用runtime给对象中的属性赋值
    /**
     class_copyIvarList: 获取类中的所有成员变量
     Ivar:成员变量
     第一个参数:表示获取哪个类中的成员变量
     第二个参数:表示这个类有多少成员变量,传入一个Int变量地址,会自动给这个变量赋值
     返回值Ivar *:指的是一个ivar数组,会把所有成员属性放在一个数组中,通过返回的数组就能全部获取到。
     count: 成员变量个数
     */
    unsigned int count = 0;
    
    //3.获取类中所有的成员变量
    Ivar *ivarList  =  class_copyIvarList(self, &count);
    
    //遍历数组所有成员变量
    for (int i = 0; i < count; i++) {
        //从数组中取出对应的成员
        Ivar ivar = ivarList[i];
        //获取成员变量的名字
        NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
        //处理成员变量 -> 字典中的key(去掉_ )
        NSString *key = [ivarName substringFromIndex:1];
        NSLog(@"ivarName:%@",key);
        //根据成员属性名去字典查找对应的value
        id value = dict[ivarName];
        
        //找到value
        if (value) {
            //模型中的属性赋值
            [objc setValue:value forKey:key];
        }
    }
    
    
    
    
    return objc;
}


@end

4.动态添加方法

如果一个类方法非常多,加载类到内存的时候比较耗费资源,需要给每个方法生产映射表,可以动态给某个类,添加方法来解决.

@implementation PersonModel


// 没有返回值,1个参数
// void,(id,SEL)
void aaa(id self, SEL _cmd, NSNumber *meter) {
    NSLog(@"跑了%@米", meter);
}

// 任何方法默认都有两个隐式参数,self,_cmd(当前方法的方法编号)
// 什么时候调用:只要一个对象调用了一个未实现的方法就会调用这个方法,进行处理
// 作用:动态添加方法,处理未实现
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    // [NSStringFromSelector(sel) isEqualToString:@"run"];
    if (sel == NSSelectorFromString(@"run:")) {
        // 动态添加run方法
        // class: 给哪个类添加方法
        // SEL: 添加哪个方法,即添加方法的方法编号
        // IMP: 方法实现 => 函数 => 函数入口 => 函数名(添加方法的函数实现(函数地址))
        // type: 方法类型,(返回值+参数类型) v:void  @:对象->self  :表示SEL->_cmd
        class_addMethod(self, sel, (IMP)aaa, "v@:@");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}


@end

//使用
    // 默认PersonModel,没有实现run:方法,可以通过performSelector调用,但是会报错。
    // 动态添加方法就不会报错
    PersonModel *person = [PersonModel new];
    [person performSelector:@selector(run:) withObject:@10];

//输出
跑了10米


参考
http://www.jianshu.com/p/19f280afcb24

上一篇 下一篇

猜你喜欢

热点阅读