美文网首页
Runtime-原理与应用

Runtime-原理与应用

作者: 攻克乃还_ | 来源:发表于2017-08-21 06:50 被阅读3次
    使用运行时前提,必须导入#import <objc/message.h>

    1.runtime使用技巧

    • 谁来发生动作,前缀就是谁。
    • 让编译器有提示:设置为NO
      TARGETS - Build Settings - 搜索msg - 改为NO
    • 查看oc底层runtime的实现
      cd 目录
      clang -rewrite-objc 文件名

    2.1.对象如何找到对应方法去调用:

    • 对象方法存放在类中,类方法存放在元类(meta class)中。
    • 类调用的方法,在runtime底层,会把类转化为类对象再调用方法。
    • 每一个类都有一个方法列表(MethodList)(已实现的方法)。
    1.根据isa指针找到所属类
    2.根据方法编号在方法列表中找到类中的方法名
    3.根据方法名到方法区找到方法的实现
    

    2.2.类如何找到类方法去调用:

    • 元类保存了类方法列表
    • 元类(meteClass)是类,也是对象
    1.根据所属类的isa指针找到元类
    2.根据方法编号在类方法列表中找到类中的方法名
    3.根据方法名到方法区找到方法的实现
    4.如果没有,向父元类查找实现
    

    3.消息机制

    • 应用场景:调用私有方法(有方法才调用,否则崩溃)
        Person *p = [[Person alloc] init];
        //不带参数 
        objc_msgSend(p, @selector(eat));
        //带参数
        objc_msgSend(p, @selector(eat), 20);
    

    4.交换方法

    • 应用场景:系统方法原本的功能可能不够用,保持原有功能扩展新的功能
    • 分类中重写load方法,在load方法中把系统原方法和新方法用runtime交换
        + (void)load
        {
        Method imageNamedMethod = class_getClassMethod(self,   @selector(imageNamed:));
        // 获取xmg_imageNamed
        Method xmg_imageNamedMethod = class_getClassMethod(self,     @selector(xmg_imageNamed:));
        // 交互方法:runtimec
        method_exchangeImplementations(imageNamedMethod,    xmg_imageNamedMethod);
        }
    

    5.添加方法

    • 应用场景:懒加载机制,有些方法很久不会调用,加载类到内存很消耗资源,动态添加方法,就是用到的时候再加载到内存。
    #import "ViewController.h"
    #import "Person.h"
    
    @implementation ViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
    
        Person *p = [[Person alloc] init];
        [p performSelector:@selector(eat)];
        [p performSelector:@selector(run:) withObject:@1];
    }
    @end
    
    #import "Person.h"
    #import <objc/message.h>
    
    @implementation Person
    
    + (BOOL)resolveInstanceMethod:(SEL)sel{
    
        if(sel == NSSelectorFromString(@"eat")){
            class_addMethod(self, sel, (IMP)eat, "v@:@");
            return YES;
        }
        if(sel == NSSelectorFromString(@"run:")){
            class_addMethod(self, sel, (IMP)(run), "v@:@");
            return YES;
        }
        return [super resolveInstanceMethod:sel];
    }
    
    void eat(id self, SEL _cmd){
        NSLog(@"吃了吗您那?");
    }
    
    void run(id self, SEL _cmd, NSNumber *num){
        NSLog(@"跑了%@米", num);
    }
    @end
    

    6.添加属性:

    • 运用场景:给系统的类添加属性
      6.1.生成分类
      6.2.重写set get方法
    @interface NSObject (Property)
    @property NSString *name;
    @end
    
    - (void)setName:(NSString *)name{
    objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
    - (NSString *)name{
    return objc_getAssociatedObject(self, @"name");
    }
    

    相关文章

      网友评论

          本文标题:Runtime-原理与应用

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