Runtime窥探 (三)| 消息转发

前言

其实,重要的不是地图,而是你的方向。所以,每条路都是一次冒险,每个地点都是一场奇遇。

蒲柳之质

消息转发机制

上回书说道消息发送中最后没有在类和父类/元类中找到方法时,程序会立即崩溃吗?答案肯定不会立即崩溃。所以才引出来这篇消息转发机制的文章,正常列表中找不到它会走到消息转发系统中......

新建类:

//--------------------- Person.h ----------------------
#import <Foundation/Foundation.h>

@interface Person : NSObject

- (void)eatWith:(NSString *)name;
- (void)happy;


//--------------------- Person.m ----------------------
#import "Person.h"
#import <objc/message.h>

@implementation Person


- (void)eatWith:(NSString *)name{
    NSLog(@"实例方法正在和%@吃饭。。。。",name);
}
- (void)happy{
    NSLog(@"实例方法正在happy。。。。");
}

我们调用:

Person *p = [Person new];
[p performSelector:@selector(sleepWith:) withObject:@"Dely"];

结果:

-[Person sleepWith:]: unrecognized selector sent to instance 0x604000031260

程序崩溃,报程序不认识这个方法,也就是找不到这个方法的实现。

为什么会崩溃?因为我们在消息发送时,在类中的methodList中没有找到这个方法列表。而消息转发机制就是处理找不到方法时的后续处理,为了更好的容错。

目的:用来处理消息的转发和调用,给对象和消息更多的机会来完成成功的调用,而不是直接crash

消息转发过程:

苹果消息转发官方文档

消息转发过程有三个阶段:

  • 1.动态方法解析
  • 2.快速消息转发
  • 3.完整方法转发

1.动态方法解析

当我们在消息发送中最后没有在类和父类/元类中找到方法时,首先会走到这个阶段。
以下方法都在NSObject中定义
对应的方法是

//方法是类方法时,会调用这个函数
+ (BOOL)resolveClassMethod:(SEL)sel;

//方法是实例方法,会调用这个函数
+ (BOOL)resolveInstanceMethod:(SEL)sel;

走到这个方法时可以利用class_addMethod给类添加方法提供了可能。

#pragma mark - --------1.动态方法解析--------

//c方法
void sleepWith(id self,SEL _cmd , NSString *name){
    NSLog(@"和%@一起睡觉",name);
}

//oc方法
- (void) sleepWith:(NSString *)name{
     NSLog(@"和%@一起睡觉",name);
}

//IMP imp 方法的实现,C方法的方法实现可以直接获得。
//如果是OC方法,可以用+ (IMP)instanceMethodForSelector:(SEL)aSelector;获得方法的实现。

//动态解析方法
+ (BOOL)resolveInstanceMethod:(SEL)sel{
    if ([NSStringFromSelector(sel) isEqualToString:@"sleepWith:"]) {
        //        class_addMethod(self, sel, (IMP)happyWith, "v@:@");
        IMP imp = [self instanceMethodForSelector:NSSelectorFromString(@"happyWith:")];
        class_addMethod(self, sel, imp, "v@:@");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

+ (BOOL)resolveClassMethod:(SEL)sel{
    return [super resolveClassMethod:sel];
}

上面Person重写了```resolveInstanceMethod````方法

入参:selector就是未被处理的方法

我们在方法中匹配了未被处理的方法名,调用class_addMethod动态添加了一个方法,来实现缺少的方法,

class_addMethod中的参数const char *types就是我们上一篇讲解的type encodings。IMP就是方法的实现变量

函数返回值:匹配到我们缺少的函数就return YES;,匹配不到就返回父类return [super resolveInstanceMethod:sel];

无论返回值为YES还是NO,都会进入下一阶段的快速消息转发。

此时再调用

Person *p = [Person new];
[p performSelector:@selector(sleepWith:) withObject:@"Dely"];

结果发现不会崩溃了:

和Dely一起睡觉

2.快速消息转发

对应的方法是

- (id)forwardingTargetForSelector:(SEL)aSelector;

这个阶段的方法运行时会询问能否把消息转给其他接收者处理,也就是此时系统给了个将这个 SEL 转给其他对象的机会。

#pragma mark - --------2.快速消息转发--------
//将消息转出给别的对象
- (id)forwardingTargetForSelector:(SEL)aSelector{
    NSLog(@"_cmd = %@",NSStringFromSelector(aSelector));
    
    if (aSelector== @selector(uppercaseString)) {
        //转给字符串对象 
        return @"person";
    }
    return [super forwardingTargetForSelector:aSelector];
}

函数返回值:函数返回消息转出的对象或者父类[super forwardingTargetForSelector:aSelector]

我们调用下面

//不会崩溃。 str = @"PERSON"
NSString *str =  [p performSelector:@selector(uppercaseString) withObject:nil];

注意:把消息转出给别的对象执行,是一对一的,也就是这个消息不能转给很多个对象来执行。

3.完整方法转发

当前面两个阶段都没有处理也可以在这个阶段来处理,
对应的方法是

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

这是消息转发的最后一个处理容错的机会。

  • methodSignatureForSelector 用于描述被转发的消息,系统会调用methodSignatureForSelector:方法,尝试获得一个方法签名。如果获取不到,则直接调用doesNotRecognizeSelector抛出异常。如果能获取,则返回非nil,然后创建一个 NSlnvocation 并传给forwardInvocation:

  • forwardInvocation 参数 anInvocation 中包含未处理消息的各种信息(selector\target\参数...)。在这个方法中,可以把 anInvocation 转发给多个对象,因为invokeWithTarget方法

#pragma mark - --------3.完整消息转发--------
//完整消息转发
-(void)forwardInvocation:(NSInvocation *)anInvocation{
    if (anInvocation.selector == @selector(testMethod)){
        ViewController *vc1 = [[ViewController alloc] init];
        ViewController *vc2 = [[ViewController alloc] init];
        [anInvocation invokeWithTarget:vc1];
        [anInvocation invokeWithTarget:vc2];
    }
}

//获取方法签名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if(aSelector == @selector(testMethod)){
        return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    }
    return nil;
} 

如果上述3个方法都没有来处理这个消息,就会进入 NSObject 的- (void)doesNotRecognizeSelector:(SEL)aSelector;方法中,抛出异常.

整个消息转发的流程图如下:

消息转发

_objc_msgForward函数

//定义
_objc_msgForward(id _Nonnull receiver, SEL _Nonnull sel, ...) 

作用:用于消息转发的。直接调用_objc_msgForward是非常危险的事,用不好会直接导致程序Crash,但是使用得当效果还是很炫的。

调用_objc_msgForward,将跳过查找 cache、Methodlist、消息转发的第一个阶段(动态消息解析) 的过程,直接触发消息转发的第2、3阶段。就算类中有这个方法,他也会告诉你没有这个方法。而直接进行消息转发的2、3阶段处理。且用且珍惜

消息转发总结

消息发送时,在类和父类/元类中没有找到方法时,通过下面的3种消息转发机制来进行容错处理。

  • 1.调用resolveInstanceMethod给个机会动态添加方法的实现
  • 2.调用forwardingTargetForSelector让消息转发给别的对象来执行
  • 3.调用forwardInvocation和methodSignatureForSelector让消息转发给别的多个对象来执行。

如果都没有处理这个消息,系统会调用doesNotRecognizeSelector抛出异常。

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

推荐阅读更多精彩内容