2022-03-09

                           iOS Runtime 消息转发机制原理

最近看到很多面试问道 runtime是个啥,咋实现的、具体实现的步骤,针对此类问题我做了一下整理,可能帮到小伙伴的可以看看。废话不多说直接走起

我们在开发过程中,经常会遇到这样的报错,当我们调用一个对象不存在的方法的时候

系统报错 提示如下错误

"-[YBHomeHeaderCRView pushMessage:]: unrecognized selector sent to instance 0x7fc77cf0de40"

其实这种报错都是iOS消息转发机制在无法响应的情况后抛出的问题。那我们来深入的看一下这个消息机制是这么走的。

这么进入的消息机制的

1.首先通过[YBHomeHeaderCRView new]对象的ISA指针找打它对应的class。

2.首先在class的cache用查找是否有sendMessage方法,没有再去class的method list 查找,找到并将其加入到cache中,方便下次调用。

3.如果没有就去它的superclass里继续查找。也是先cache,在methodlists。

4.一旦查找到对应的函数方法,通过函数指针IMP调转到对应的函数中去执行。

5.如果一直没有找到,就执行消息转发。

消息机制原理

我们先看一下下面这个结构图,先对整个消息处理机制有一个初步的认识

iOS消息转发机制

从结构图来看,消息转发机制共分为3大步骤:

1.Method resolution 动态方法解析处理阶段

2.Fast forwarding 快速转发阶段

3.Normal forwarding 慢速转发阶段

如果想要不抛出unrecognized selector 的报错,也就需要从这3步里面来做补救。

第一步 动态方法解析处理阶段

如果调用了对象方法首先会进行+(BOOL)resolveInstanceMethod:(SEL)sel判断

如果调用了类方法 首先会进行 +(BOOL)resolveClassMethod:(SEL)sel判断

如果YES则能接受消息,NO则不能接受消息 则就回进入第二步

我们先调一下对象方法,然后在resolveInstanceMethod进行补救,

void pushMessage(id self, SEL _cmd, NSString *msg){

    NSLog(@"------%@",msg);

}

+(BOOL)resolveInstanceMethod:(SEL)sel{

    NSString *methodName =NSStringFromSelector(sel);

    if ([methodName isEqualToString:@"pushMessage:"]) {

        return class_addMethod(self, sel, (IMP)pushMessage, "v@:@");

    }

    return  NO;

}

经过上面类型的补救,果然对象方法不在抛出异常了,并且打印了数据

1

第二歩:快速转发阶段

如果在上一步的方法内返回的为YES则能接受消息 NO不能接受消息 则进入第二步,我们先把上面方法内的处理方案注释掉,让消息转发进入第二步。

我们新创建一个YBMessage类,里面声明和实现pushMessage方法,用来当作备用响应者。

-(id)forwardingTargetForSelector:(SEL)aSelector{

    NSString *methodName =NSStringFromSelector(aSelector);

    if ([methodName isEqualToString:@"pushMessage:"]) {

        return [YBMessage new];

    }

    return [super forwardingTargetForSelector:aSelector];

}

因为一个对象内部可能还有其他可能响应的对象,所以这个方法是转发SEL去对象内部的其他可以响应该方法的对象。

第三歩 慢速转发阶段

如果第2步返回self或者nil,则说明没有可以响应的目标 则进入第三步。第三步的消息转发机制本质上跟第二步是一样的都是切换接受消息的对象,但是第三步切换响应目标更复杂一些,需要手动将响应方法切换给备用响应对象。

-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{

    NSString*methodName =NSStringFromSelector(aSelector);

    if([methodNameisEqualToString:@"pushMessage:"]) {

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

    }

    return [super methodSignatureForSelector:aSelector];

}

-(void)forwardInvocation:(NSInvocation *)anInvocation{

    SELsel = [anInvocationselector];

    YBMessage *msg = [YBMessage new];

    if([msg isEqualToString:sel]) {

        [anInvocationinvokeWithTarget:msg];

    }else{

        [superforwardInvocation:anInvocation];

    }

}

//例子"v@:@"

//v@:@ v:返回值类型void;@ id类型,执行sel的对象;:SEL;@参数

如果前三个都没有找到方法

- (void)doesNotRecognizeSelector:(SEL)aSelector{

    NSLog(@"找不到方法");

}

同时,越往后面处理代价越高,最好的情况是在第一步就处理消息,这样runtime会在处理完后缓存结果,可以提高处理效率。

总结:

在一个函数找不到时,OC提供了三种方式去补救:

1、调用resolveInstanceMethod给个机会让类添加这个实现这个函数

2、调用forwardingTargetForSelector让别的对象去执行这个函数

3、调用forwardInvocation(函数执行器)灵活的将目标函数以其他形式执行。

如果都不中,调用doesNotRecognizeSelector抛出异常。

参考 iOS Runtime 消息转发机制原理和实际用途

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

推荐阅读更多精彩内容