iOS 消息发送那些事儿

拜读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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,762评论 0 9
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 776评论 0 1
  • 参考链接: http://www.cnblogs.com/ioshe/p/5489086.html 简介 Runt...
    乐乐的简书阅读 2,153评论 0 9
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 760评论 0 2
  • 本文详细整理了 Cocoa 的 Runtime 系统的知识,它使得 Objective-C 如虎添翼,具备了灵活的...
    lylaut阅读 822评论 0 4