iOS消息转发

我们已经研究了objc_msgSend从汇编快速查找缓存流程,慢速查找流程,动态方法决议流程,如果这几个流程下来都没找到合适的执行方法,接下来就会走到消息转发流程。
消息转发流程都有哪些呢?我们创建一个mac项目,在main.m执行下面代码

extern void instrumentObjcMessageSends(BOOL flag);

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"Hello, World!");
        
        SJPerson *p = [SJPerson alloc];
        instrumentObjcMessageSends(true);
        [p say1];
        instrumentObjcMessageSends(NO);
    }
    return 0;
}

say1方法不实现,会报错,然后在finder中前往文件夹输入路径:/tmp/msgSends。找到一个msgSends-xxxx的文件,打开,会发现代码执行流程:

image

从这里面可以看到resolveInstanceMethod,还有一系列其他方法,后面的就是消息转发流程所调用的方法。

这个函数是怎么来的呢?从源码中log_and_fill_cache方法找到的。

static void
log_and_fill_cache(Class cls, IMP imp, SEL sel, id receiver, Class implementer)
{
#if SUPPORT_MESSAGE_LOGGING
    if (slowpath(objcMsgLogEnabled && implementer)) {
        bool cacheIt = logMessageSend(implementer->isMetaClass(), 
                                      cls->nameForLogging(),
                                      implementer->nameForLogging(), 
                                      sel);
        if (!cacheIt) return;
    }
#endif
    cls->cache.insert(sel, imp, receiver);
}

这里面logMessageSend方法就是记录相关日志信息,方法里面有这个日志的路径:"/tmp/msgSends-%d"
所以我们只需要让if判断为真就可以记录相关日志。
implementer一般情况下都是有值的,这时就关注objcMsgLogEnabled就可以了。全局搜索,找到这个变量赋值的地方在下面这个函数:

void instrumentObjcMessageSends(BOOL flag)
{
    bool enable = flag;

    // Shortcut NOP
    if (objcMsgLogEnabled == enable)
        return;

    // If enabling, flush all method caches so we get some traces
    if (enable)
        _objc_flush_caches(Nil);

    // Sync our log file
    if (objcMsgLogFD != -1)
        fsync (objcMsgLogFD);

    objcMsgLogEnabled = enable;
}

所以只需要重新声明一下这个函数并调用,就可以生成相关日志信息给我们参考了。

快速转发流程

- (id)forwardingTargetForSelector:(SEL)aSelector
{
    NSLog(@"%s", __func__);
    if (aSelector == @selector(say1)) {
        return [SJStudent alloc];
    }
    return nil;
}

快速转发流程,就是返回一个可以接收消息的对象,当返回值是nil或者这个对象也无法处理这个消息时,会走消息慢速转发流程。

慢速转发流程

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

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    NSLog(@"sel : %s,   target:%@", anInvocation.selector, anInvocation.target);
}

慢速转发流程有两个方法,必须成套使用。第一个需要返回一个方法签名,这个签名里面方法 的typeEncoding不一定要跟当前方法的typeEncoding完全一致,只要类似就行。只实现第一个方法还会崩溃,第二个方法实现后不会崩溃。

动态决议、消息转发方法走两遍原因探究

方法的动态决议,消息转发对应的方法,如果我们不做任何处理,只打印方法信息,就会发现每个方法都会执行两遍

+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    SJLog(@"%s", __func__);
    return [super resolveInstanceMethod:sel];
}

- (id)forwardingTargetForSelector:(SEL)aSelector
{
    SJLog(@"%s", __func__);
    return [super forwardingTargetForSelector:aSelector];
}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    SJLog(@"%s", __func__);
    return [super methodSignatureForSelector:aSelector];
}

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    SJLog(@"%s", __func__);
}
image.png

因为方法动态决议可以动态添加方法,添加完后会继续走消息转发流程,如果消息转发流程走完发现没有处理,系统会主动又调用一次这个方法,所以就会走两次。

方法动态决议意义

NSObject添加resolveInstanceMethod,会发现无论对象方法还是类方法,最后都会走到NSObject的这个方法里面,我们可以在这个方法统一处理。
这样项目里所有的方法找不到,我们都能统一监听。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容

  • 上一篇我们梳屡了消息的发送和查找流程,我们会发现对应类方法和对象方法的动态解析是分开的,这是为什么呢,原因两点: ...
    海浪萌物阅读 1,072评论 0 0
  • 在iOS - 方法查找流程一文中,提到过当查找不到方法时会进行动态方法决议,如果动态方法决议也找不到该怎么办呢?那...
    e521阅读 673评论 0 1
  • runtime方法查找流程及消息转发 方法查找 方法查找的流程:缓存查找-->当前类查找-->父类逐级查找 1.缓...
    雷海洋阅读 1,010评论 0 2
  • 在上篇文章方法查找流程通过在类和父类的缓存以及方法列表中进行查找,如果直到查找到NSObject中都没有找到,然后...
    瞬间完善阅读 281评论 0 1
  • 消息转发(Message Forwarding)是在查找 IMP 失败后执行一系列转发流程的慢速通道,如果不作转发...
    二猪哥阅读 3,883评论 0 7