runtime 消息调用机制

2016-08-21  本文已影响40人  JihanWen

方法调用流程

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)]的方式调用方法
对与①情况编译器会直接报错,而对于②情况需要到运行时才能确定对象能否接收指定的消息,这时候会进入下面所说的消息转发的流程;

消息转发

struct objc_method { SEL method_name ;//方法名为此方法的签名 char *method_types ;//方法类型描述了参数的类型。 IMP method_imp ;//函数指针,为方法具体实现代码块的地址 }

消息转发.png

+ (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; }

(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

上一篇 下一篇

猜你喜欢

热点阅读