美文网首页
转:Runtime的使用

转:Runtime的使用

作者: 为零sowill | 来源:发表于2017-02-06 15:34 被阅读10次

    一、交换两个方法的实现,拦截系统自带的方法调用功能

    需要用到的方法 <objc/runtime.h>
    获得某个类的类方法
    Method class_getClassMethod(Class cls , SEL name)
    获得某个类的实例对象方法
    Method class_getInstanceMethod(Class cls , SEL name)
    交换两个方法的实现
    void method_exchangeImplementations(Method m1 , Method m2)

    案例1:方法简单的交换

    创建一个Person类,类中实现以下两个类方法,并在.h 文件中声明

    + (void)run {
        NSLog(@"跑");
    }
    
    + (void)study {
        NSLog(@"学习");
    }
    

    控制器中调用,则先打印跑,后打印学习

    [Person run];
    [Person study];
    

    下面通过runtime 实现方法交换,类方法用class_getClassMethod ,对象方法用class_getInstanceMethod

    // 获取两个类的类方法
    Method m1 = class_getClassMethod([Person class], @selector(run));
    Method m2 = class_getClassMethod([Person class], @selector(study));
    // 开始交换方法实现
    method_exchangeImplementations(m1, m2);
    // 交换后,先打印学习,再打印跑!
    [Person run];
    [Person study];
    

    案例2:拦截系统方法

    需求:比如iOS6 升级 iOS7 后需要版本适配,根据不同系统使用不同样式图片(拟物化和扁平化),如何通过不去手动一个个修改每个UIImage的imageNamed:方法就可以实现为该方法中加入版本判断语句?

    步骤:
    1、为UIImage建一个分类(UIImage+Category)
    2、在分类中实现一个自定义方法,方法中写要在系统方法中加入的语句,比如版本判断

    + (UIImage *)xh_imageNamed:(NSString *)name {
        double version = [[UIDevice currentDevice].systemVersion doubleValue];
        if (version >= 7.0) {
            // 如果系统版本是7.0以上,使用另外一套文件名结尾是‘_os7’的扁平化图片
            name = [name stringByAppendingString:@"_os7"];
        }
        return [UIImage xh_imageNamed:name];
    }
    

    3、分类中重写UIImage的load方法,实现方法的交换(只要能让其执行一次方法交换语句,load再合适不过了)

    + (void)load {
        // 获取两个类的类方法
        Method m1 = class_getClassMethod([UIImage class], @selector(imageNamed:));
        Method m2 = class_getClassMethod([UIImage class], @selector(xh_imageNamed:));
        // 开始交换方法实现
        method_exchangeImplementations(m1, m2);
    }
    

    注意:自定义方法中最后一定要再调用一下系统的方法,让其有加载图片的功能,但是由于方法交换,系统的方法名已经变成了我们自定义的方法名(有点绕,就是用我们的名字能调用系统的方法,用系统的名字能调用我们的方法),这就实现了系统方法的拦截!
    利用以上思路,我们还可以给 NSObject 添加分类,统计创建了多少个对象,给控制器添加分类,统计有创建了多少个控制器,特别是公司需求总变的时候,在一些原有控件或模块上添加一个功能,建议使用该方法!

    二、在分类中设置属性,给任何一个对象设置属性

    众所周知,分类中是无法设置属性的,如果在分类的声明中写@property 只能为其生成get 和 set 方法的声明,但无法生成成员变量,就是虽然点语法能调用出来,但程序执行后会crash,有人会想到使用全局变量呢?比如这样

    int _age;
    
    - (int )age {
        return _age;
    }
    
    - (void)setAge:(int)age {
        _age = age;
    }
    

    但是全局变量程序整个执行过程中内存中只有一份,我们创建多个对象修改其属性值都会修改同一个变量,这样就无法保证像属性一样每个对象都拥有其自己的属性值。这时我们就需要借助runtime为分类增加属性的功能了

    需要用到的方法 <objc/runtime.h>

    set方法,将值value 跟对象object 关联起来(将值value 存储到对象object 中)
    参数 object:给哪个对象设置属性
    参数 key:一个属性对应一个Key,将来可以通过key取出这个存储的值,key 可以是任何类型:double、int 等,建议用char 可以节省字节
    参数 value:给属性设置的值
    参数policy:存储策略 (assign 、copy 、 retain就是strong)
    void objc_setAssociatedObject(id object , const void *key ,id value ,objc_AssociationPolicy policy)

    利用参数key 将对象object中存储的对应值取出来
    id objc_getAssociatedObject(id object , const void *key)

    步骤:

    1、创建一个分类,比如给任何一个对象都添加一个name属性,就是NSObject添加分类(NSObject+Category)
    2、先在.h 中@property 声明出get 和 set 方法,方便点语法调用

    @property(nonatomic,copy)NSString *name;

    3、在.m 中重写set 和 get 方法,内部利用runtime 给属性赋值和取值

    char nameKey;
    
    - (void)setName:(NSString *)name {
        // 将某个值跟某个对象关联起来,将某个值存储到某个对象中
        objc_setAssociatedObject(self, &nameKey, name, OBJC_ASSOCIATION_COPY_NONATOMIC);
    }
    
    - (NSString *)name {
        return objc_getAssociatedObject(self, &nameKey);
    }
    

    相关文章

      网友评论

          本文标题:转:Runtime的使用

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