美文网首页
iOS 消息发送那些事儿

iOS 消息发送那些事儿

作者: wufeifan890330 | 来源:发表于2016-07-06 16:55 被阅读141次

    拜读yulingtianxia大神关于Runtime的博文后,针对消息发送这一块,做点小记录。

    参考链接:Objective-C Runtime

    objective-c中,[target selector] 在运行时会被Runtime系统转化为objc_msgSend函数。而该函数内其实做了一系列复杂的操作。

    一.检查target是否为nil,如果是nil则忽略。

    (objective-c中,任意一个nil对象发送消息都不会崩溃)

    二.查找方法的实现

    1.先从缓存中查找;【objective-c中的对象在运行时会被Runtime系统转化为结构体。在结构体中,有一个isa指针,它最终指向的就是对象所属的类。isa实际是一个指向objc_class结构体的指针,该结构体下关联了它的超类指针,类名,成员变量,方法,缓存,还有附属的协议。这里我们要说的就是缓存。当某个方法被调用之后,Runtime就会把该方法保存到cache中。】

    2.从方法分发表中查找,找不到就顺着父类的分发表继续查找,直到NSObject;

    3.分发表找不到则进入动态方法解析

    三、动态方法解析

    Runtime系统会调用resolveInstanceMethod: 或 resolveClassMethod:方法来给我们一次为类动态添加方法的机会。当这两个方法返回NO时,则进入消息转发。

    +(BOOL)resolveInstanceMethod:(SEL)aSEL

    {

    if(aSEL==@selector(test)){

    // // 该方法第一个参数为添加方法的类

    // // 第二个参数为方法的声明

    // // 第三个参数为方法的实现

    // // 第四个参数为方法的类型即TypeEncodings。从左到右依次为返回值类型、参数类型。可参考官方文档https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtTypeEncodings.html

    // class_addMethod([self class], aSEL, (IMP)test, "v@:");

    // return YES;

    returnNO;

    }

    return[superresolveInstanceMethod:aSEL];

    }

    上述代码中,if语句内如果返回YES,则整个消息发送的过程到此为止。真正执行的就是test内部的实现。

    四、消息转发

    1.消息转发之前,可以通过-(id)forwardingTargetForSelector:(SEL)aSelector 实现重定向 如果该方法返回nil或者self,则进入消息转发机制

    -(id)forwardingTargetForSelector:(SEL)aSelector

    {

    if(aSelector==@selector(test)){

    // // 这里我在另一个类中声明了一个test的方法用以测试

    // return [Test02 new];

    returnself;

    }

    return[superforwardingTargetForSelector:aSelector];

    }

    上述代码中,如果返回的是注释部分的代码,则消息发送过程到此为止,消息将被转发至[UIApplication sharedApplication].delegate对象。(从头开始消息发送过程)。

    2.通过重写 forwardInvocation: 来实现转发逻辑。实际上在forwardInvocation: 消息发送之前,Runtime系统会向对象发送 methodSignatureForSelector: 来获取返回的方法签名用于生成NSInvocation对象。 所以在重写forwardInvocation: 方法的同时也需要重写methodSignatureForSelector: 方法,并且返回正确的方法签名。否则会抛出无法识别方法的异常。

    // 消息转发

    -(void)forwardInvocation:(NSInvocation*)anInvocation

    {

    if(anInvocation.selector==@selector(test)){

    [anInvocation invokeWithTarget:[Test02new]];

    }else{

    [superforwardInvocation:anInvocation];

    }

    }

    // 返回方法签名

    -(NSMethodSignature*)methodSignatureForSelector:(SEL)aSelector

    {

    // return [NSMethodSignature signatureWithObjCTypes:"v@:"];

    // 签名对象记录着方法的参数和返回值的类型信息。

    NSMethodSignature*signature=[supermethodSignatureForSelector:aSelector];

    if(!signature){

    // 本例中,因为要将消息转发给Test02对象,因此我们直接实例化一个Test02对象,并获取其同名方法的签名,并返回

    signature=[[Test02new]methodSignatureForSelector:aSelector];

    }

    returnsignature;

    // return nil;

    }

    注:动态添加方法,消息转发时,最好确保参数及返回值等信息一致。

    如上述代码中,若获取方法签名的方法返回nil则抛出无法识别方法的异常;

    若返回的签名信息不符,签名信息多余仍然可以执行,只是有可能发生“灵异事件”。若签名信息较少,则会抛出-[NSInvocation getArgument:atIndex:]: index (1) out of bounds [-1, 0]的错误。因为在调用方法时,从签名信息中获取参数和返回值信息就会出现越界情况了。

    附下载链接—测试的小demo

    相关文章

      网友评论

          本文标题:iOS 消息发送那些事儿

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