Runtime-原理与应用

使用运行时前提,必须导入#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");
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,776评论 0 9
  • 对于从事 iOS 开发人员来说,所有的人都会答出【runtime 是运行时】什么情况下用runtime?大部分人能...
    梦夜繁星阅读 3,733评论 7 64
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,643评论 33 466
  • 目录 Objective-C Runtime到底是什么 Objective-C的元素认知 Runtime详解 应用...
    Ryan___阅读 1,961评论 1 3
  • 影片一開始,幾米《向左走•向右走》的故事情節閃過腦海,隨之被片頭曲驅散——這曲風轉換得讓耳朵有點跟不上節奏。 整部...
    蘿卜阅读 371评论 2 1