美文网首页
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 消息调用机制

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