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消息转发
第一步,调用一个对象的方法,进入对象方法列表中寻找,找到执行。找不到进入父类寻找。直到找到,找不到进入下一步。
第二步,给开发者一个机会 动态的给对象添加一个方法去执行。若不处理进入下一步.
第三步,进入本步说明这个对象不处理此方法的调用,这时可以让开发者指定一个能处理此方法的对象。形象的说明就是 这件事小明没能力做,但任务已经派发了,他认为小刚能做,就把这个任务让小刚去做。若不处理进入下一步。
第四步,进入这一步说明问题很大了,不过系统还是给开发者最后一个机会,重新指定一个人,指定一个任务,重新包装一下去执行。就好像这件事没人管没人执行,但任务下来了,得找个应急的临时工顶一下,随便做点啥都行,让领导知道有人干活就好。如果临时工也没有找,那肯定要出大事了。系统崩溃、闪退、报错。