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 ;//函数指针,为方法具体实现代码块的地址 }

消息转发.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; }

  • 阶段三
    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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,762评论 0 9
  • 继上Runtime梳理(四) 通过前面的学习,我们了解到Objective-C的动态特性:Objective-C不...
    小名一峰阅读 769评论 0 3
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,231评论 0 7
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,592评论 33 466
  • 少熬夜 少熬夜 少熬夜 少熬夜 少熬夜 少熬夜 少熬夜 少熬夜啊~~~
    黑与白的故事阅读 201评论 0 0