RunTime 之消息处理与消息转发

前言

有关Runtime的知识总结,我本来想集中写成一篇文章的,但是最后发现实在是太长,而且不利于阅读,最后分成了如下几篇:


OC方法的调用其实是消息的发送,

消息的发送其实是C语言函数的调用

在Runtime中不得不提的就是OC的消息处理和消息转发机制。我们知道在OC中的实例对象调用一个方法称作消息传递,OC中里的消息传递采用动态绑定机制来决定具体调用哪个方法,OC的实例方法在转写为C语言后实际就是一个函数,但是OC并不是在编译期决定调用哪个函数,而是在运行期决定。

OC究竟是怎么将实例方法转换为C语言的函数,又是如何调用这些函数的呢?这些都依靠强大的runtime。
在深入代码之前介绍一个clang编译器的命令:

clang -rewrite-objc main.m

该命令可以将.m的OC文件转写为.cpp文件
有如下代码:

int main(int argc, const char * argv[]) {
@autoreleasepool {
     //为了方便查看转写后的C语言代码,将alloc和init分两步完成
    Person *p = [Person alloc];
    p = [p init];
    p.name = @"Jiaming Chen";
    [p showMyself];
  }
  return 0;
}

通过上述clang命令可以转写代码,然后找到如下定义:

int main(int argc, const char * argv[]) {
/* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool;

    Person *p = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("Person"), sel_registerName("alloc"));
    p = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)p, sel_registerName("init"));
    ((void (*)(id, SEL, NSString *))(void *)objc_msgSend)((id)p, sel_registerName("setName:"), (NSString *)&__NSConstantStringImpl__var_folders_1f_dz4kq57d4b19s4tfmds1mysh0000gn_T_main_f5b408_mi_1);
    ((void (*)(id, SEL))(void *)objc_msgSend)((id)p, sel_registerName("showMyself"));

   }
  return 0;
}

OC语句被 clang编译器转化为 C语言后的样子。

当你调用一个类的方法时:

(1)先在本类中的方法缓存列表中进行查询,如果在缓存列表中找到了该方法的实现,就执行,
(2)如果找不到就在本类中的方列表中进行查找。在本类方列表中查找到相应的方法实现后就进行调用,
(3)如果没找到,就去父类中进行查找。如果在父类中的方法列表中找到了相应方法的实现,那么就执行,

否则就执行消息处理与消息转发相关的方法。

总结一下流程图就是如下:


这里要特别说明下:

A:在这个resolveInstanceMethod 方法中,这个函数是给类利用class_addMethod添加函数的机会。根据文档,如果实现了添加函数代码则返回YES,未实现返回NO。如果没有添加函数代码就算返回YES,也无任何意义,还是会往下走。

B:返回的这个对象,如果也无法响应这个 SEL就会跟当前这个情形一样,会调用这个对象的resolveInstanceMethod 方法,如果未作处理也会帮抛出异常。

C: methodSignatureForSelector:(SEL)aSelector 这个函数让重载方有机会抛出一个函数的签名,再由后面的forwardInvocation:去执行。
forwardInvocation: 在这个函数里可以将NSInvocation多次转发到多个对象中,这也是这种方式灵活的地方。(forwardingTargetForSelector只能以Selector的形式转向一个对象)

下面具体介绍下相关方法的使用:

一、消息处理(Resolve Method)
  • 首先,如果沿继承树没有搜索到相关方法则会向接收者所属的类进行一次请求,看是否能够动态的添加一个方法。

  • 当在相应的类以及父类中找不到类方法实现时会执行+resolveInstanceMethod:这个类方法。该方法如果在类中不被重写的话,默认返回NO。

  • 在该方法中,我们可以为找不到实现的SEL动态的添加一个方法实现,添加完毕后,就会执行我们添加的方法实现。这样,当一个类调用不存在的方法时,就不会崩溃了。

在Model的父类、BaseViewController的父类中实现如下方法,
可以避免调用没有实现的方法造成的崩溃。

#import "MyObject.h"
#import "NSObject+NoMethod.h"

@implementation MyObject
/** 没有找到SEL时会执行下面的方法 */
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    [self addMethod:sel methodImp:@selector(dynamicAddMethod)];
    return YES;
}

- (void)dynamicAddMethod
{
    NSLog(@"找不到方法时执行这里");
}
@end
二、消息单项转发

如果不对上述消息进行处理的话,也就是+resolveInstanceMethod:返回NO时,会走下一步消息转发,即-forwardingTargetForSelector:

该方法会返回一个类的对象,这个类的对象有SEL对应的实现,当调用这个找不到的方法时,就会被转发到SecondClass中去进行处理。这也就是所谓的消息转发。当该方法返回self或者nil, 说明不对相应的方法进行转发,那么就该走下一步了。

- (id)forwardingTargetForSelector: (SEL)aSelector
{
      //  return [[Test alloc]init];
          return nil;
      //  return self;
})
三、消息多项转发

如果不将消息转发给其他类的对象,那么就只能自己进行处理了、或者崩溃。会用到如下两个方法:

  - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
  - (void)forwardInvocation:(NSInvocation *)anInvocation

(1) -methodSignatureForSelector:方法来获取方法的参数以及返回数据类型,也就是说该方法获取的是方法的签名并返回。

注意:如果 methodSignatureForSelector 返回的NSMethodSignature 是 nil 的话不会继续执行 forwardInvocation,转发流程终止,抛出无法处理的异常。

methodSignatureForSelector如何实现?

methodSignatureForSelector用于描述被转发的消息,描述的格式要遵循以下规则点击打开链接

首先,先要了解的是,每个方法都有self和_cmd两个默认的隐藏参数,self即接收消息的对象本身,_cmd即是selector选择器,所以,描述的大概格式是:返回值@:参数。@即为self,:对应_cmd(selector).返回值和参数根据不同函数定义做具体调整。

比如下面这个函数

-(void)testMethod; 
返回值为void,没有参数,按照上面的表格中的符号说明,再结合上面提到的概念,这个函数的描述即为   v@:

v代表void,@代表self(self就是个对象,所以用@),:代表_cmd(selector)

如果实在拿不准,不会写,还可以简单写段代码,借助method_getTypeEncoding方法去查看某个函数的描述,比如

-(NSString *)testMethod2:(NSString *)str;
描述为 @@:@

-(void)testMethod
{
    Method method = class_getInstanceMethod(self.class, @selector(testMethod));
    const char *des = method_getTypeEncoding(method);
    NSString *desStr = [NSString stringWithCString:des encoding:NSUTF8StringEncoding];
    NSLog(@"->   %@",desStr);
}
-(NSString *)testMethod2:(NSString *)str
{
    Method method = class_getInstanceMethod(self.class, @selector(testMethod2:));
    const char *des = method_getTypeEncoding(method);
    NSString *desStr = [NSString stringWithCString:des encoding:NSUTF8StringEncoding];
    NSLog(@"->   %@",desStr);
    return @"";
}

把数字去掉,剩下v@: ,与之前我们的描述一致。
结果是@@:@ ,与之前结论一致。

(2) 这个时候runtime会将未知消息的所有细节都封装为NSInvocation对象,然后调用下述方法:

- (void)forwardInvocation: (NSInvocation*)invocation;

在这个函数里可以将NSInvocation多次转发到多个对象中。

调用这个方法如果不能处理就会调用父类的相关方法,一直到NSObject的这个方法,如果NSObject都无法处理就会调用doesNotRecognizeSelector:方法抛出异常。

综合运用如下:

@implementation Test
-(void)func3;
{
    NSLog(@"KK: %s",__func__);
}
@end
@implementation Test2
-(void)func3;
{
    NSLog(@"KK: %s",__func__);
}
@end


@implementation Person

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    if(aSelector == @selector(func3))
    {
        return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    }
    NSMethodSignature *new= [super methodSignatureForSelector:aSelector];
    return new;
}

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    if (anInvocation.selector == @selector(func3))
    {
        Test *h1 = [[Test alloc] init];
        Test2 *h2 = [[Test2 alloc] init];
        [anInvocation invokeWithTarget:h1];
        [anInvocation invokeWithTarget:h2];
    }
}

[person func3];

person中没有实现func3方法,但最终运行后,程序没有报错,且Test 和 Test2 的func3 方法都被执行了。

了解更多 NSInvocation

An NSInvocation is an Objective-C message rendered static, that is, it is an action turned into an object. NSInvocation objects are used to store and forward messages between objects and between applications, primarily by NSTimer objects and the distributed objects system.

NSInvocation有点类似于java里的反射,它有一套完整的装备:target,selector,returnValue,ArgumentArray,有了它们,NSInvocation就可以动态的invoke任意对象的任意方法了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容

  • 建议你从今以后准备一个资料夹用来收集些标题的精选范例,以便你在构思自己的营销素材时用作参考.假如你一时想不出广告标...
    文字扑克林奇Lynch阅读 2,077评论 0 0
  • 很多年前有几个人总是喜欢说我 心思深沉犹疑不定看不透我 更多的人坚信我有很多面孔 每次我孤独地坐着抱着孤独的枕头 ...
    年轮止阅读 245评论 2 2
  • 蒲公英模式也称“分红交易”, 是由留学归国的杨甦宏博士(现任浙江 123 图书馆馆长)开创的用于图书交易、流通和循...
    1a4f74b75da0阅读 429评论 0 1
  • 完成了一个demo,多少有点成就感。
    左壮右悦阅读 90评论 0 0
  • 你听过那么多道理,却依然经营不好自己的人生。 但是,道理又哪是看过就会明白的呢。很多事,很多人,不经历过,不撞过南...
    青木___阅读 133评论 0 0