runtime 消息调用机制
- 在对象调用方法是Objective-C中经常使用的功能,也就是消息的传递,而Objective-C是C的超集,所以和C不同的是,Objective-C使用的是动态绑定,也就是runtime。
方法调用流程
1、编译器会把 [self doSomething] 转化objc_msgSend(ViewController,SEL),SEL为@selector(doSomething)。
2、Runtime会在self对象所对应的ViewController类的方法缓存列表里查找方法的SEL(cache)
3、如果没有找到,则在ViewController类的方法分发表查找方法的SEL。(类由对象isa指针指向,方法分发表即method_list)
4、如果没有找到,则在其父类的方法分发表里查找方法的SEL
(父类由类的superClass指向)
5、如果没有找到,则沿继承体系继续下去,最终到达NSObject类。
6、如果在2345的其中一步中找到,则定位了方法实现的入口,执行具体实现
7、如果还是没找到那就会面临两种情况:
① 如果是使用[self doSomething]的方式调用方法 ② 使用[self performSelector:@selector(doSomething)]的方式调用方法
对与①情况编译器会直接报错,而对于②情况需要到运行时才能确定对象能否接收指定的消息,这时候会进入下面所说的消息转发的流程;
消息转发
-
1.IMP是”implementation”的缩写,它是objetive-C 方法(method)实现代码块的地址,可像C函数一样直接调用。通常情况下我们是通过[object method:parameter]或objc_msgSend()的方式向对象发送消息,然后Objective-C运行时寻找匹配此消息的IMP,然后调用它
-
2.Objetive-C中的Method结构
在Objecitve-C中,在类中对每一个方法有一个在运行时构建的数据结构,在Objective-C 2.0中,此结构对用户不可见,但仍在内部存在。
struct objc_method { SEL method_name ;//方法名为此方法的签名 char *method_types ;//方法类型描述了参数的类型。 IMP method_imp ;//函数指针,为方法具体实现代码块的地址 }
- 阶段一
+ (BOOL)resolveInstanceMethod:(SEL)sel { if (sel == @selector(doSomething)) { NSLog(@"add method here"); class_addMethod([self class],sel, (IMP)dynamicMethodIMP,"v@:"); return YES; } return [super resolveInstanceMethod:sel]; }
- 阶段二
这时候已经默许了你并不想使用消息接收者来响应这个方法,所以我们需要找到一个接盘侠
- (id)forwardingTargetForSelector:(SEL)aSelector { Class class=NSClassFromString(@"BBViewController"); UIViewController *vc = class.new; if (aSelector == NSSelectorFromString(@"secondVCMethod")){ NSLog(@"secondVC do this !"); return vc; } return nil; }
- 阶段三
runtime需要生成一个methodSignature变量来组装,这将通过调用消息接收者的-(NSMethodSignature *)methodSignatureForSelector:
获取,这个变量包含了方法的参数类型、参数个数以及消息接收者等信息。接着把这个变量组装成一个NSInvocation对象进行最后一次的消息转发,调用接收者的-forwardInvocation: 来进行最后的挽救机会
(void)forwardInvocation:(NSInvocation *)anInvocation { Class class=NSClassFromString(@"BBViewController"); UIViewController *vc = class.new; if ([class instancesRespondToSelector:anInvocation.selector]) { [anInvocation invokeWithTarget:vc]; } }
参考文章
http://www.cocoawithlove.com/2008/02/imp-of-current-method.html
延伸
https://github.com/bang590/JSPatch/blob/master/README-CN.md