美文网首页
runtime 消息调用机制

runtime 消息调用机制

作者: JihanWen | 来源:发表于2016-08-21 16:22 被阅读40次
  • 在对象调用方法是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

相关文章

  • 深入浅出Runtime

    Runtime运行时 目录 一、runtime 简介二、消息机制<了解>2.1 消息机制原理2.2 消息调用流程三...

  • runtime的实用性讲解

    runtime 概念 runtime 运行时机制,主要是消息机制。OC 中的方法调用属于消息的发送,动态调用过程,...

  • 方法调用底层实现

    runtime怎么实现方法的调用 :消息机制,runtime系统会把方法调用转化为消息发送。即objc-msgSe...

  • iOS面试-Runtime简介

    本文主要介绍runtime的五点 Runtime简介 Runtime(消息机制) Runtime方法调用流程 Ru...

  • runtime 消息调用机制

    在对象调用方法是Objective-C中经常使用的功能,也就是消息的传递,而Objective-C是C的超集,所以...

  • iOS Runtime学习笔记(一) - 基础学习

    iOS Runtime学习笔记 Runtime就是运行时, 核心就是消息机制. 对OC的函数调用,是一个动态调用过...

  • runtime解析及常用方法

    什么是runtime? runtime直译:运行时机制;OC发送消息的本质,就是 runtime去调用苹果底层的一...

  • oc 中的runtime 消息机制(二)

    runtime消息机制 消息发送 在Object-C中,我们其实可以直接调用C的代码也就是Runtime的C语...

  • runtime原理及应用

    runtime简介 Runtime就是运行时, 核心就是消息机制. 对OC的函数调用,是一个动态调用过程,只有在运...

  • 深入浅出Runtime (二) Runtime的消息机制

    Runtime消息机制 消息发送 在Object-C中,我们其实可以直接调用C的代码也就是Runtime的C语言代...

网友评论

      本文标题:runtime 消息调用机制

      本文链接:https://www.haomeiwen.com/subject/jfjujttx.html