×

iOS Runtime 小试(1)

96
YM_1
2016.03.01 15:27* 字数 619

runtime在平时简单的开发中很少用到,但这个涉及到底层,做iOS的不懂runtime就等于功夫只学了招式没有修炼内功,最多只能是个高手,懂的api多点,但运行机制不通。api学不完,但底层运行机制是不变的。所有今天主要学习了下runtime。自己写的代码
我先看了一个视频 iOS runtime 分享 总结下里面讲了两个知识点。
1 交换方法实现

+ (void)load{
    
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Class cls = [UILabel class];
        SEL original = @selector(willMoveToSuperview:);
        SEL swizzled = @selector(myWillMoveToSuperview:);
        
        Method originalMethod   = class_getInstanceMethod(cls, original);
        Method swizzledMethod   = class_getInstanceMethod(cls, swizzled);
        
        //添加自定义的方法到类里面
        BOOL didAddMethod = class_addMethod(cls, original, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod));
/**
         swizzled 是存在于 UILabel 的方法列表中
         original 是存在于 UILabel 的父类方法列表中,不存在在UILabel中。 
         所以class_addMethod 是给UILabel添加一个选择器,相当于重写了父类的方法。
         重写的方法名,追加另外一个函数的实现。
         */
        if (didAddMethod) {
            //方法已经添加成功  替换掉原来的方法实现
            class_replaceMethod(cls, swizzled, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod));
        }else{
            //没有添加成功    改变两个函数的实现
            method_exchangeImplementations(originalMethod, swizzledMethod);
        }
    });
}

- (void)myWillMoveToSuperview:(nullable UIView *)newSuperview{
    
    [self myWillMoveToSuperview:newSuperview];
    if (newSuperview) {
        self.backgroundColor = [UIColor yellowColor];
        [self setFont:[UIFont systemFontOfSize:13]];
    }
}

method_exchangeImplementations 改变两个方法的实现
很多人可能会认为上面的myWillMoveToSuperview:会循环执行,其实不会的。我来捋一捋。交换地址后 系统会调用willMoveToSuperview,可这时候willMoveToSuperview 指向了 myWillMoveToSuperview,于是就执行

 {
 [self myWillMoveToSuperview:newSuperview];
    if (newSuperview) {
        self.backgroundColor = [UIColor yellowColor];
        [self setFont:[UIFont systemFontOfSize:13]];
    }
}

这时候代码调用 myWillMoveToSuperview 其实是调用的

  [self willMoveToSuperview:newSuperview];

然后接下来执行判断等代码。
这里面有很多弯不好绕,要静下心好好捋一下。


2 消息转发

//c语言的函数  Obj-C的方法(method)就是一个至少需要两个参数(self,_cmd)的C函数
//Since the function must take at least two arguments—self and _cmd, the second and third characters must be “@:” (the first character is the return type).
void minus(id self,SEL _cmd,NSNumber *val){
    NSLog(@"%0.2f",[[self valueForKey:@"value"] floatValue] - [val floatValue]);
}


//第一步      没有方法 添加方法实现
+ (BOOL)resolveInstanceMethod:(SEL)sel{
    if (sel == @selector(minus:)) {
        //v@:f     返回值为void  @: the second and third characters must be “@:”
        class_addMethod([self class], sel, (IMP) minus, "v@:f");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

//第二步   指向新的方法接收者
- (id)forwardingTargetForSelector:(SEL)aSelector{
    if (aSelector == @selector(uppercaseString)) {
        return @"hello world";
    }
    return [super forwardingTargetForSelector:aSelector];
}

简单介绍下消息转发。轻松学习之一--Objective-C消息转发


第一步,调用一个对象的方法,进入对象方法列表中寻找,找到执行。找不到进入父类寻找。直到找到,找不到进入下一步。


第二步,给开发者一个机会 动态的给对象添加一个方法去执行。若不处理进入下一步.


第三步,进入本步说明这个对象不处理此方法的调用,这时可以让开发者指定一个能处理此方法的对象。形象的说明就是 这件事小明没能力做,但任务已经派发了,他认为小刚能做,就把这个任务让小刚去做。若不处理进入下一步。


第四步,进入这一步说明问题很大了,不过系统还是给开发者最后一个机会,重新指定一个人,指定一个任务,重新包装一下去执行。就好像这件事没人管没人执行,但任务下来了,得找个应急的临时工顶一下,随便做点啥都行,让领导知道有人干活就好。如果临时工也没有找,那肯定要出大事了。系统崩溃、闪退、报错。


runtime
Web note ad 1