iOS Runtime 小试(1)

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


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


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


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


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


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 目录 Objective-C Runtime到底是什么 Objective-C的元素认知 Runtime详解 应用...
    Ryan___阅读 5,946评论 1 3
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,810评论 33 466
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 5,901评论 0 9
  • Runtime是什么 Runtime 又叫运行时,是一套底层的 C 语言 API,其为 iOS 内部的核心之一,我...
    SuAdrenine阅读 4,444评论 0 3
  • 一、Runtime简介 Runtime简称运行时。OC就是运行时机制,也就是在运行时候的一些机制,其中最主要的是消...
    林安530阅读 4,695评论 0 2

友情链接更多精彩内容