Runtime:objc_msgSend执行流程

目录
一,基本流程
二,消息发送
三,动态方法解析
四,消息转发

一,基本流程

1,方法的调用都会转换为objc_msgSend函数的调用,通常称为消息机制

// OC代码
[person eat];
// 底层代码(用clang进行转换)
((void (*)(id, SEL))(void *)objc_msgSend)((id)person, sel_registerName("eat"));
// 简化代码
objc_msgSend(person, @selector(eat)); // person为消息接收者,eat为消息名称

2,objc_msgSend执行流程

执行流程
二,消息发送

1,底层代码(源码下载地址

IMP lookUpImpOrForward(Class cls, SEL sel, id inst, 
                       bool initialize, bool cache, bool resolver)
{
    IMP imp = nil;
    bool triedResolver = NO;

 retry:    
    runtimeLock.assertLocked();

    // Try this class's cache.
    // 查找自身的缓存
    imp = cache_getImp(cls, sel);
    if (imp) goto done;

    // Try this class's method lists.
    // 查找自身的方法列表
    {
        Method meth = getMethodNoSuper_nolock(cls, sel);
        if (meth) {
            // 找到就放入自身的缓存中
            log_and_fill_cache(cls, meth->imp, sel, inst, cls);
            imp = meth->imp;
            goto done;
        }
    }

    // Try superclass caches and method lists.
    // 查找父类的缓存和方法列表
    {
        unsigned attempts = unreasonableClassCount();
        // 遍历父类
        for (Class curClass = cls->superclass;
             curClass != nil;
             curClass = curClass->superclass)
        {
            // Halt if there is a cycle in the superclass chain.
            if (--attempts == 0) {
                _objc_fatal("Memory corruption in class list.");
            }
            
            // Superclass cache.
            // 查找父类的缓存
            imp = cache_getImp(curClass, sel);
            if (imp) {
                if (imp != (IMP)_objc_msgForward_impcache) {
                    // Found the method in a superclass. Cache it in this class.
                    // 找到就放入自身的缓存中
                    log_and_fill_cache(cls, imp, sel, inst, curClass);
                    goto done;
                }
                else {
                    // Found a forward:: entry in a superclass.
                    // Stop searching, but don't cache yet; call method 
                    // resolver for this class first.
                    break;
                }
            }
            
            // Superclass method list.
            // 查找父类的方法列表
            Method meth = getMethodNoSuper_nolock(curClass, sel);
            if (meth) {
                // 找到就放入自身的缓存中
                log_and_fill_cache(cls, meth->imp, sel, inst, curClass);
                imp = meth->imp;
                goto done;
            }
        }
    }

    // 这里省略了动态方法解析和消息转发的代码

 done:
    runtimeLock.unlock();

    return imp;
}

2,图解

消息发送

3,说明

  • 如果调用实例方法,就在class对象的缓存和方法列表中查找;如果调用类方法,就在meta-class对象中查找

  • 方法列表指的是class_rw_t里的methods

  • 如果方法列表已排序,就用二分查找法;如果没有排序,就用顺序查找法

三,动态方法解析

1,实例代码

// Person
@interface Person : NSObject
- (void)eat;
@end

@implementation Person
- (void)run {
    NSLog(@"Person run");
}
// 添加实例方法需实现此方法,添加类方法需实现resolveClassMethod
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    if (sel == @selector(eat)) {
        Method runMethod = class_getInstanceMethod(self, @selector(run));
        // 动态添加eat的实现
        class_addMethod(self,
                        @selector(eat),
                        method_getImplementation(runMethod),
                        method_getTypeEncoding(runMethod));
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}
@end

// 使用
Person *person = [Person new];
[person eat];

// 打印
Person run

2,底层代码

IMP lookUpImpOrForward(Class cls, SEL sel, id inst,
                       bool initialize, bool cache, bool resolver)
{
    IMP imp = nil;
    bool triedResolver = NO;
    
    // 这里省略了消息发送的代码

    // No implementation found. Try method resolver once.
    // triedResolver为NO,未动态方法解析
    if (resolver  &&  !triedResolver) {
        runtimeLock.unlock();
        // 调用resolveInstanceMethod或resolveClassMethod
        resolveMethod(cls, sel, inst);
        runtimeLock.lock();
        // Don't cache the result; we don't hold the lock so it may have
        // changed already. Re-do the search from scratch instead.
        // 标记已动态方法解析
        triedResolver = YES;
        // 重新执行消息发送
        goto retry;
    }
    
    // 这里省略了消息转发的代码

 done:
    runtimeLock.unlock();

    return imp;
}

3,图解


动态方法解析

4,说明

  • 手动实现resolveInstanceMethodresolveClassMethod,并且在其中动态添加方法,是动态方法解析的关键

  • 动态添加的方法会存入class_rw_t里的methods中,所以需要重新执行消息发送

  • 在消息发送过程中能否找到方法取决于是否有动态添加方法

四,消息转发

1,不开源

IMP lookUpImpOrForward(Class cls, SEL sel, id inst,
                       bool initialize, bool cache, bool resolver)
{
    IMP imp = nil;
    bool triedResolver = NO;
    
    // 这里省略了消息发送和动态方法解析的代码
    
    // No implementation found, and method resolver didn't help.
    // Use forwarding.

    // 无法查看内部具体实现
    imp = (IMP)_objc_msgForward_impcache;
    cache_fill(cls, sel, imp, inst);

 done:
    runtimeLock.unlock();

    return imp;
}

2,第一个阶段

  • 实例代码
// Person
@interface Person : NSObject
- (void)eat;
@end

@implementation Person
- (id)forwardingTargetForSelector:(SEL)aSelector {
    if (aSelector == @selector(eat)) {
        // 将eat转发给Dog去处理
        Dog *dog = [Dog new];
        return dog;
    }
    return [super forwardingTargetForSelector:aSelector];
}
@end

// Dog
@interface Dog : NSObject
- (void)eat;
@end

@implementation Dog
- (void)eat {
    NSLog(@"Dog eat");
}
@end

// 使用
Person *person = [Person new];
[person eat];

// 打印
Dog eat
  • 伪底层代码
if (class_respondsToSelector(receiverClass, @selector(forwardingTargetForSelector:))) {
    id forwardingTarget = [receiver forwardingTargetForSelector:sel];
    if (forwardingTarget && forwardingTarget != receiver) {
        // forwardingTarget是返回的dog,sel是eat
        return objc_msgSend(forwardingTarget, sel, ...);
    }
}

3,第二阶段

  • 实例代码
@implementation Person
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if (aSelector == @selector(eat)) {
        // 返回eat的方法签名,告知系统需要转发eat,系统就会调用forwardInvocation
        return [[Dog new] methodSignatureForSelector:@selector(eat)];
    }
    return [super methodSignatureForSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
    // 将eat转发给Dog去处理
    Dog *dog = [Dog new];
    [anInvocation invokeWithTarget:dog];
}
@end

// 打印
Dog eat
  • 伪底层代码
if (class_respondsToSelector(receiverClass, @selector(methodSignatureForSelector:))) {
    // 调用methodSignatureForSelector
    NSMethodSignature *methodSignature = [receiver methodSignatureForSelector:sel];
    if (methodSignature && class_respondsToSelector(receiverClass, @selector(forwardInvocation:))) {
        NSInvocation *invocation = [NSInvocation _invocationWithMethodSignature:methodSignature frame:frameStackPointer];
        // 调用forwardInvocation
        [receiver forwardInvocation:invocation];
        void *returnValue = NULL;
        [invocation getReturnValue:&value];
        return returnValue;
    }
}

4,图解

消息转发

5,说明

  • forwardInvocation只要实现了就不会抛出异常,即使在该方法中什么也不做

  • 如果转发的是类方法,需要将上面三个方法改成类方法,另外target也要设置为类对象

// 类方法
+ (id)forwardingTargetForSelector:(SEL)aSelector {
    if (aSelector == @selector(eat)) {
        // 类对象
        return [Dog class];
    }
    return [super forwardingTargetForSelector:aSelector];
}
  • NSInvocation是对将被转发消息的封装
// Person
@interface Person : NSObject
- (int)eat:(int)food;
@end

@implementation Person
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if (aSelector == @selector(eat:)) {
        return [[Dog new] methodSignatureForSelector:@selector(eat:)];
    }
    return [super methodSignatureForSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
    NSLog(@"消息接收者---%@", anInvocation.target);
    NSLog(@"消息---%@", NSStringFromSelector(anInvocation.selector));
    int argument;
    [anInvocation getArgument:&argument atIndex:2]; // 系统默认会添加两个参数
    NSLog(@"参数---%d", argument);
    int returnValue;
    [anInvocation getReturnValue:&returnValue];
    NSLog(@"返回值---%d", returnValue);
}
@end

// 使用
Person *person = [Person new];
[person eat:1];

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

推荐阅读更多精彩内容

  • 文中的实验代码我放在了这个项目中。 以下内容是我通过整理[这篇博客] (http://yulingtianxia....
    茗涙阅读 924评论 0 6
  • 前言 想要通过runtime发送消息,就必须要掌握runtime如何发送消息,是调用哪个函数?又是如何调用的?本篇...
    G_GUI阅读 639评论 0 0
  • 消息发送阶段:负责从类及父类的缓存列表及方法列表查找方法。 动态解析阶段:如果消息发送阶段没有找到方法,则会进入动...
    大白菜s阅读 332评论 0 0
  • 本文详细整理了 Cocoa 的 Runtime 系统的知识,它使得 Objective-C 如虎添翼,具备了灵活的...
    lylaut阅读 804评论 0 4
  • 什么是树 树是一种类似于链表的数据结构,不过链表的结点是以线性方向简单地指向其后继结点,而树的一个结点可以指向许多...
    不知名的蛋挞阅读 498评论 0 0