runtime
runtime 就是运行时
C语言声明了方法是一定要实现的,不然编译的部分就会给报错。OC的方法声明完调用后不进行实现,也是可以编译过去的。但是运行的时候才会给报错。
xcode将消息机制的代码提示给关闭了,这里需要自己手动去打开才可以有代码提示。buildSetting -> msg -> enable strict checking of objc_msgSend calls -> no
OC的代码,在底层会被转化为消息发送。
[self walk];
在底层就会被更改为
objc_msgSend(self, @selector(walk));
KVC、KVO的底层都是使用了runtime,内存检测工具也是使用了runtime。
runtime的基本用法
self.persion = [Persion new];
self.persion.name = @"Tom";
- 使用runtime改变变量的值
// 实例变量个数
unsigned int count = 0;
// 获取persion中的所有属性变量
Ivar* ivar = class_copyIvarList([self.persion class], &count);
// 遍历
for (int i = 0; i < count; i++) {
// 实例变量
Ivar var = ivar[i];
// 返回一个C字符串的变量名称
const char* varName = ivar_getName(var);
// 转换成NSString类型
NSString* name = [NSString stringWithUTF8String:varName];
if ([name isEqualToString:@"_name"]) {
object_setIvar(self.persion, var, @"Jerry");
break;
}
}
- 使用runtime交换方法
// 获取实例方法
Method m1 = class_getInstanceMethod([self.persion class], @selector(firstMethod));
Method m2 = class_getInstanceMethod([self.persion class], @selector(secondMethod));
// 交换
method_exchangeImplementations(m1, m2);
- 使用runtime添加方法
/// 动态添加方法
- (void) addMethod {
//v@:@ v -> void @ -> id : -> SEL @ -> mile
class_addMethod([self.persion class], @selector(run:), (IMP)runMethod, "v@:@");
}
void runMethod(id self, SEL _cmd, NSString* miles) {
NSLog(@"%@", miles);
}
测试添加的方法成功了没有
if ([self.persion respondsToSelector:@selector(run:)]) {
[self.persion performSelector:@selector(run:) withObject:@"9 miles"];
} else {
NSLog(@"方法没有找到");
}
- 使用runtime添加属性
添加属性需要加一个category
#import "Persion+mult.h"
#import <objc/runtime.h>
@implementation Persion (mult)
const char* name1 = "jom";
/// 动态为分类扩展属性
- (void)setNick:(NSString *)nick {
objc_setAssociatedObject(self, &name1, nick, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
- (NSString*) nick {
return objc_getAssociatedObject(self, &name1);
}
@end
调用
self.persion.nick = @"cat";
NSLog(@"%@", self.persion.nick);