iOS消息转发流程

一、前言

一个类对象查找方法,我们都知道是先从缓存列表中去查找,然后在去方法列表里查找,这样就能快速的查找到相关的imp,但是当我们没有查找到相应的imp时,系统又会做一些什么事情呢?带着这样的好奇我们开始源码的探究,我们知道如果一个方法没有实现,运行时是会崩溃并且报错;如下所示:

Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '+[LGPerson sayNB]: unrecognized selector sent to class 0x1000022a8'

二、动态方法解析

但是当我们在源码的动态方法解析过程做相应的事情时:程序就不会报错,可以继续执行并且进行我们想要的执行结果;其中核心代码来自resolveMethod_locked(inst, sel, cls, behavior);

if (slowpath(behavior & LOOKUP_RESOLVER)) {
behavior ^= LOOKUP_RESOLVER;
return resolveMethod_locked(inst, sel, cls, behavior);
}

1代码调试

我们在项目中创建一个对象LGPerson 继承自 NSObject,其中包括了几个简单的方法;

@interface LGPerson : NSObject

- (void)sayNB;
- (void)sayMaster;
- (void)say666;
- (void)sayHello;

+ (void)sayNB;
+ (void)lgClassMethod;

@end

而实现只包括以下几个方法

- (void)sayHello{
    NSLog(@"%s",__func__);
}

- (void)sayNB{
    NSLog(@"%s",__func__);
}
- (void)sayMaster{
    NSLog(@"%s",__func__);
}


+ (void)lgClassMethod{
    NSLog(@"%s",__func__);
}

也就是说- (void)say666;+ (void)sayNB; 只有声明。没有实现。我们在main.m 文件中进行相关的方法调用,此时必然会出现文章开头的崩溃现象,

LGPerson *person = [LGPerson alloc];
[person say666];

带着问题的探索进入到动态方法解析的方法探索resolveInstanceMethodresolveClassMethod,看看究竟在动态解析过程中做了什么事情;
方法中我们只是简单的打印了一句方法的名称来到这里;

+ (BOOL)resolveInstanceMethod:(SEL)sel{
    NSLog(@"%@ 来了",NSStringFromSelector(sel));
    return [super resolveInstanceMethod:sel];
   
}

此时我们看到控制台仍然会崩溃,但是我们打印的方法日志会出现在崩溃之前,


动态方法日志打印.png

这里打印两次;也就是说这个动态方法解析的过程中resolveInstanceMethod 会调用两次 ;为什么会走两次接下来会分析;

图片.png

根据条件判断,我们看到当前的behavior = 3,而 LOOKUP_RESOLVER = 2 在根据计算slowpath(2) 进入我们的第一次动态方法解析;动态方法解析的源码如下,由于我们是调用的对象方法,所以走图中标注的地方,

图片.png

动态方法的实现如下;

   runtimeLock.assertUnlocked();
    ASSERT(cls->isRealized());
    SEL resolve_sel = @selector(resolveInstanceMethod:);

    // lookup resolveInstanceMethod
    if (!lookUpImpOrNil(cls, resolve_sel, cls->ISA())) {
        // Resolver not implemented.
        return;
    }
    BOOL (*msg)(Class, SEL, SEL) = (typeof(msg))objc_msgSend;
    bool resolved = msg(cls, resolve_sel, sel);
    IMP imp = lookUpImpOrNil(inst, sel, cls);

所以第一次动态方法解析的sel == resolveInstanceMethod,然后将执行后的imp返回,再次进行方法查找流程。所以此处会执行两次。

接着在resolveInstanceMethod中简单的进行一个处理如下;

+ (BOOL)resolveInstanceMethod:(SEL)sel{
    NSLog(@"%@ 来了",NSStringFromSelector(sel));

    if (sel == @selector(say666)) {
        NSLog(@"%@ 来了",NSStringFromSelector(sel));

        IMP imp           = class_getMethodImplementation(self, @selector(sayMaster));
        Method sayMMethod = class_getInstanceMethod(self, @selector(sayMaster));
        const char *type  = method_getTypeEncoding(sayMMethod);
        return class_addMethod(self, sel, imp, type);
    }
    return [super resolveInstanceMethod:sel];
   
}

此时就会出现一个神奇的现象。程序正常执行。这就是动态方法解析的作用;

2 原理解析;

动态方法解析的源码截图如上:分为两种,一种是对象,另一种是类;

  • 1 如果是对象方法,则在当前类对象中查找,因为对象方法就存储在当前类对象的方法列表中
  • 2 如果是类方法,先从本类中查找是否有该类的imp,如果找到就直接返回,如果还是没有找到,再对该类的元类,根元类NSObject中查询,如果有就返回,没有就进行消息转发流程
  • 3 消息转发 也分类两种情况,一种是快速转发,一种是慢速转发流程;

三、快速转发流程

1 日志辅助

上面我们已经知道了相关的消息处理流程,我们接下来分析一下相关的消息快速转发流程,我们借助了instrumentObjcMessageSends 进行相关的日志跟踪;代码如下;

extern void instrumentObjcMessageSends(BOOL flag);

int main(int argc, const char * argv[]) {
    @autoreleasepool {

        LGPerson *person = [LGPerson alloc];
        instrumentObjcMessageSends(YES);
        
        [person sayInstanceMethod];
        
        instrumentObjcMessageSends(NO);


        NSLog(@"Hello, World!");
    }
    return 0;
}

再次进入会到相应的位置


日志地址.png

此时编译项目我们找到相应的日志文件;/tmp/msgSends 目录下会多出一个文件;

图片.png

打开文件,里边有相应的方法执行信息;


日志打印.png

从以上我们能发现执行的流程是resolveInstanceMethod ,再到forwardingTargetForSelector,继而到methodSignatureForSelector,这就是以上的 动态方法解析 > 消息快速转发 > 消息慢速转发的完整流程。

2 概念的介绍;

以上得知我们先要进行消息的快速转发,也就是forwardingTargetForSelector 我们在开源的代码中搜索并没有找到相应的定义。借此我们借助苹果官方文档进行查询;

Returns the object to which unrecognized messages should first be directed.
这就是苹果官方的定义,也就是找到该方法的第一实现者,从而就就结束了,
所以此时我们

-(id)forwardingTargetForSelector:(SEL)aSelector
{
    return [LGStudent alloc];
    
}

此时在运行代码就不会出现崩溃的情况了,因为该方法内已经查询到我们调用方法的实现 ,也就是不管是不是我们自己方法接受者实现的,只要有这个方法的imp 程序就不会崩溃。

四,慢速转发流程

如果快速查找流程没有实现,那么就进入慢速查找流程的过程;那么就进入慢速转发的流程;
methodSignatureForSelector,慢速转发流程搭载forwardInvocation 来使用;

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
    
    return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    
}

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    anInvocation.target = [LGStudent alloc];
    [anInvocation invoke];
    
}

此时我们同样也能得到相应的程序正常执行流程,这就是慢速转发的过程,
通过NSInvocation 我们能看到相关的定义如下

图片.png

所以此处我们不仅仅能修改target 还能监控其他的属性;

图片.png

通过控制台我们能监控相关的NSInvocation 属性,从而动态的修改相关内容也能起到慢速转发的作用和目的。

五、总结

以上就是对象调用方法的整套消息转发流程,从而实现消息转发,从创建对象,到查找对象的isa,在进入类的bits_t,以及相关的类的结构,缓存以及imp的查找,到最后的方法转发机制,这就是一个对象在iOS开发系统的存在和意义,通过以上的相关学习和文章的整理,自己也对相关的消息机制进行一个深刻的认识,这就是成长路上的一点小小的进步吧,继续努力。

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