美文网首页
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 消息发送那些事儿

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

  • iOS 消息发送与转发详解

    iOS 消息发送与转发详解 iOS 消息发送与转发详解

  • iOS 截图的那些事儿

    iOS 截图的那些事儿 iOS 截图的那些事儿

  • IOS消息传递机制

    ios的消息传递机制分为三个阶段:消息发送阶段,动态解析阶段,消息转发阶段。 消息发送阶段: 当ios的对象调用方...

  • Runtime那些事儿(消息机制)

    Runtime那些事儿(消息机制) Runtime那些事儿(消息机制)

  • iOS面试题总结(二)

    iOS面试题(二) 消息发送和转发机制,SEL和IMP 消息发送转载自黄龙辉消息发送和消息转发机制 在Object...

  • iOS runtime学习(二)

    iOS runtime学习(一) 1、发送消息 方法调用的本质,就是让对象发送消息。 objc_msgSend,只...

  • vue3.0 前端和客户端混合开发

    1.客户端发送消息 iOS给前端发送消息 vue3.0中接收客户端发送消息(在index.html中) 前端接收客...

  • iOS 发送位置消息

    发送地理位置在社交应用里面是很常用的需求。最近也需要在自己的应用里面加入这个功能 首先我们需要获取自己的地理位置,...

  • iOS消息发送流程

    上一篇我们讲到从objc_msgSend发送消息进入到了汇编,然后通过CacheLookup---> CheckM...

网友评论

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

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