使用运行时前提,必须导入#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");
}