美文网首页
iOS Runtime 小试(1)

iOS Runtime 小试(1)

作者: YM_1 | 来源:发表于2016-03-01 15:27 被阅读174次

    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消息转发


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


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


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


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


    相关文章

      网友评论

          本文标题:iOS Runtime 小试(1)

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