美文网首页iosiOS开发大圆满集合iOS开发实用技术
个人对iOS OC 消息转发机制的基本原理理解

个人对iOS OC 消息转发机制的基本原理理解

作者: Hedgehog___ | 来源:发表于2018-04-09 18:56 被阅读1722次

        关于OC的消息转发机制,是大部分面试官在面试过程中经常问到的问题。在此我整理了一下我对OC消息转发机制的理解。

        众所周知OC的一个对象在发送消息的时候首先在该类的struct objc_method_list列表中去搜索,如果找到则直接调用相关方法的实现,如果没有找到就会通过super_class指针沿着继承树向上去搜索,如果找到就跳转,如果到了继承树的根部(通常为NSObject)还没有找到。那就会调用NSObjec的一个方法doesNotRecognizeSelector:,这样就会报unrecognized selector 错误。其实在调用doesNotRecognizeSelector:方法之前还会进行消息转发---还有三次机会来补救。也就是常说的OC消息转发的三次补救措施。

        总的来说一个OC消息的发送会经历四个阶段(该四个阶段都是搜索到NSObject再进入下阶段)

        1)先在本类中搜索改方法的实现,如果有则直接调用若果没有则去父类中搜索直到NSObject,如果NSObject没有则进入消息转发(类的动态方法解析、备用接受者对象、完整的消息转发)。

        2)类的动态方法解析:

        首先创建SonPerson类,在ViewController 里面写

          id person = [[SonPerson alloc]init];

        [person appendString:@""];

        注意这里要用id 不然编译报错。

        在该类和父类中没有找到该方法的实现则会执行 +(BOOL)resolveClassMethod:(SEL)sel 或+(BOOL)resolveInstanceMethod:(SEL)sel 方法。在+(BOOL)resolveClassMethod:(SEL)sel 或+(BOOL)resolveInstanceMethod:(SEL)sel 方法 中利用黑魔法runtime 动态添加方法实现。

void dynamicAdditionMethodIMP(id self,SEL _cmd){

    NSLog(@"dynamicAdditionMethodIMP");

}

+(BOOL)resolveClassMethod:(SEL)sel{

    NSLog(@"resolveInstanceMethod: %@", NSStringFromSelector(sel));

    if(sel ==@selector(appendString:)) {

        class_addMethod([selfclass], sel, (IMP)dynamicAdditionMethodIMP,"v@:");

        returnYES;

    }

    return[superresolveClassMethod:sel];

}

+(BOOL)resolveInstanceMethod:(SEL)sel{

    NSLog(@"resolveInstanceMethod: %@", NSStringFromSelector(sel));

    if(sel ==@selector(appendString:)) {

        class_addMethod([selfclass], sel, (IMP)dynamicAdditionMethodIMP,"v@:");

        returnYES;

    }

    return[super resolveInstanceMethod:sel];

}

BOOL class_addMethod(Class cls, SEL name, IMP imp,constchar*types);

第一个参数是需要添加方法的类,第二个参数是一个selector,也就是实例方法的名字,第三个参数是一个IMP类型的变量也就是函数实现,需要传入一个C函数,这个函数至少有两个参数,一个是id self一个是SEL _cmd,第四个参数是函数类型。具体设置方法可以看注释。

控制台输出:

resolveInstanceMethod: appendString:

dynamicAdditionMethodIMP

    2)备用接受者: 在+(BOOL)resolveClassMethod:(SEL)sel 或+(BOOL)resolveInstanceMethod:(SEL)sel 方法返回NO的时候进入备用接受者阶段 。

    创建一个备用接受者类ForwardPerson 实现appendString:方法

    -(void)appendString:(NSString*)str{

        NSLog(@"%@===%@",NSStringFromClass([self class]),NSStringFromSelector(_cmd));

    }

    在SonPerson类中实现- (id)forwardingTargetForSelector:(SEL)aSelector 方法 并返回一个备用接受者对象 

- (id)forwardingTargetForSelector:(SEL)aSelector{

NSLog(@"forwardingTargetForSelector");

return [ForwardPerson new];

}

控制台输出:

forwardingTargetForSelector

 ForwardPerson===appendString:

    3)完整的消息转发:当-(void)forwardInvocation:(NSInvocation*)anInvocation 方法方法nil的时候则进入消息转发的最后阶段,完整的消息转发。也需要创建一个转发对象ForwardInvocation 

#import "ForwardInvocation.h"

@implementationForwardInvocation

-(void)appendString:(NSString*)str{

    NSLog(@"%@===%@",NSStringFromClass([self class]),NSStringFromSelector(_cmd));

}

@end

    在SonPerson中实现-(void)forwardInvocation:(NSInvocation*)anInvocation和- (NSMethodSignature*)methodSignatureForSelector:(SEL)aSelector方法

-(void)forwardInvocation:(NSInvocation*)anInvocation{

    NSLog(@"forwardInvocation");

    if ([ForwardInvocation instancesRespondToSelector:anInvocation.selector]) {

        [anInvocation invokeWithTarget:self.invocation];

    }

}

/*必须重新这个方法,消息转发机制使用从这个方法中获取的信息来创建NSInvocation对象 返回nil上面方法不执行*/

- (NSMethodSignature*)methodSignatureForSelector:(SEL)aSelector{

    NSMethodSignature*signature = [super methodSignatureForSelector:aSelector];

    if(!signature){

        if ([ForwardInvocation instancesRespondToSelector:aSelector]){

            signature = [ForwardInvocation instanceMethodSignatureForSelector:aSelector];

        }

    }

    returnsignature;

}

控制台输出:

forwardInvocation

ForwardInvocation===appendString:

最后附Demo:GitHub - SionChen/OBJC_SendMsg  并附消息转发一张图:

相关文章

  • 消息转发机制原理?

    iOS runtime探究(二): 从runtime开始深入理解OC消息转发机制https://www.jians...

  • iOS理解Objective-C中消息转发机制附Demo

    iOS理解Objective-C中消息转发机制附Demo iOS理解Objective-C中消息转发机制附Demo

  • 个人对iOS OC 消息转发机制的基本原理理解

    关于OC的消息转发机制,是大部分面试官在面试过程中经常问到的问题。在此我整理了一下我对OC消息转发机制的理解。 ...

  • iOS消息转发机制个人理解笔记

    iOS开发过程中,有一类的错误会经常遇到,就是找不到所调用的方法,当然这类问题比较好解决,给当前对象或其父类对象添...

  • 关于Runtime 消息发送机制的延伸

    说到OC 不得不说一下OC 的消息转发机制;何为OC 的消息转发机制;其实就是这样的; Objc 在向一个对象发送...

  • runtime系列文章总结

    《iOS Runtime详解(消息机制,类元对象,缓存机制,消息转发)》《消息转发机制与Aspects源码解析》《...

  • iOS中的消息转发机制

    OC由于运行时特性,可以在运行期间动态添加方法,这个寻找动态添加的方法的过程就是动态消息转发。iOS的消息转发机制...

  • OC消息机制,消息转发机制

    Runtime简称运行时,其中最主要的是消息机制 概述 C 与 OC 的不同 1.C 语言,函数的调用在编译的时候...

  • iOS 底层原理 - 消息转发

    在上一篇 iOS 底层原理 - 消息查找流程中,我们知道OC消息机制分为三个阶段,消息发送,动态解析和消息转发,如...

  • 2021iOS面试题

    iOS开发组件化; 说下对runloop的理解; 项目中遇到的难点; 事件的响应链机制; 消息的传递及消息的转发机...

网友评论

  • 萧城x:少了 resolveInstanceMethod
  • 99998cf47c68:如果是未实现的类方法不能在resolveClassMethod中直接使用class_addMethod添加方法,我又在resolveClassMethod中又调了resolveInstanceMethod,在resolveInstanceMethod中实现的方法替换
  • 动感超人丶:消息转发机制的实际应用场景呢?
  • Hanser0503:我想问下在第一个方法和第三个方法里面对这个类添加addappend方法有什么不同?
  • c281c994fc12:作者了解下 cache,每次会先cache 再遍历(散列表)
    Hedgehog___:是么?我了解下

本文标题:个人对iOS OC 消息转发机制的基本原理理解

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