iOS runtime 动态添加属性,方法,方法交换案例

2019-12-18  本文已影响0人  孙掌门

iOS runtime 动态添加属性,方法,方法交换案例

动态添加属性

我之前写过分类的文章,说过,分类不能添加成员变量,虽然我们这样去写了@interface TestClass (runtime) @property(nonatomic,copy)NSString *name; @end

但是只是生成了对应的set和get方法,并没有生成对应的成员变量,需要通过runtime的关联对象的方式给对象添加成员变量, objc_setAssociatedObject,并没有直接添加到宿主类上面,而是由一个 AssociationManager , 管理在一个 hashMap 当中,他是一个全局的容器,为不同的类添加关联对象,也都放在这个map当中,

/*
     加入下面是为某个类添加的成员变量
     */
    
    {
        /*
         类1:
         @property(copy)NSString *name;
         */
        // 类1
        // 最外面的 key 为被关联的对象,value 是一个 ObjcAssociationMap ,ObjcAssociationMap 中,@selector 为被关联的值得 key,policy 为策略,是 retain还是copy 等
        "0x1242342432":{
            "@selector(name)":{
                "value":"sunchengxiu",
                "policy":"copy"
            }
        },
        // 类2
        "0x54354253223":{
            
            "@selector(age)":{
                "value":"25"
                "policy":"retain"
            }
        }
    }
    

类似于上面的json,最外层表示为哪个对象添加了关联对象,然后用关联对象的名字作为key,value是一个对象,里面存放着值和修饰符。


@implementation TestClass (runtime)
static const char *key = "name";
-(NSString *)school{
    return objc_getAssociatedObject(self, key);
    
}
-(void)setSchool:(NSString *)school{
    
    objc_setAssociatedObject(self, key, school, OBJC_ASSOCIATION_COPY);
}
@end

那么以上的关联对象正是用了 runtime 的动态添加属性的方式。

动态添加方法

上篇讲过,runtime 消息转发的第一步,动态方法解析,我们可以动态添加一个方法,通过 runtime


void  toPrint(id obj,SEL  _cmd){
    printf("to test resolveInstanceMethod");
}
+(BOOL)resolveInstanceMethod:(SEL)sel{
    if (sel == @selector(print)) {
        class_addMethod([self class], sel, (IMP)toPrint, "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

这里就是动态添加了一个方法

方法交换

比如我们经常做 http 请求,那么我们想每次都打印这个url的值和判断是否为空应该怎么做呢?这时候就用到了,动态方法交换


@implementation NSURL (runtime)
+(void)load{
    Method m1 = class_getClassMethod([NSURL class], @selector(URLWithString:));
    Method m2 = class_getClassMethod([NSURL class], @selector(scx_URLWithString:));
    method_exchangeImplementations(m1, m2);
}
+(instancetype)scx_URLWithString:(NSString *)URLString{
    NSURL *url = [NSURL scx_URLWithString:URLString];
    
    if (!url) {
        NSLog(@"url 空了");
    } else {
        NSLog(@"%@",url);
    }
    return url;
}
@end

如上面的代码,这样我们就做到了,动态方法交换,就可以每次都判断这个url是否为空了。我们要在系统调用方法之前进行交换,所以要卸载 +load 里面。当我们调用系统的 URLWithString 其实就是调用了我们的 scx_URLWithString, 而我们在这个方法里面有调用scx_URLWithString了这个,其实就是调用系统的URLWithString,所以不会产生死循环。

上一篇下一篇

猜你喜欢

热点阅读