Runtime和它的消息机制

Objective-C这门语言,众所周知,是对C进行了扩展,具体来说进行了两个方面的扩展,面向对象的特性和smalltalk中的消息传递。而消息传递机制归根结底是建立在Runtime库上。正是这种机制,决定了Objective-C是一门动态语言,而同样是对C扩展的C++,是静态的。Objective-C将很多决定性的操作依靠Runtime在运行时处理,而C++仅仅在编译时就决定了如何处理

消息传递

在我们所熟悉的调用方法背后,最终都是以消息传递的方式进行处理,例如[object method],从表面上来看,是object调用了method方法,实际上,在运行时,是给object发送了一条method消息,这条消息不一定非要object来处理,也可以转发给其他的对象处理,也可以不进行处理,这些种种操作,都是利用Runtime在运行时处理的。
对于[object method]的调用,编译器会将其编译成一行C语言的函数:

objc_msgSend(object, @selector(method));
消息传递的步骤

了解了消息传递之后,需要进一步知道消息传递的具体步骤:

  1. 先查看method方法是不是需要被忽略
  2. 查看object对象是否为nil(Objective-C中允许空对象调用任何方法的原因)
  3. 查看缓存中是否存在方法,系统把近期发送过的消息记录在其中,苹果认为这样可以提高效率
  4. 如果缓存中没有命中,那么查找该类的方法表,依次从后往前查找
  5. 如果没有找到,则进入父类查找
  6. 如果到了根类还是没有找到的话,那么就进入动态解析
Runtime中的基本类型

以上过程虽然读起来蛮容易理解的,但是我们还得搞清楚Runtime是通过什么进行上述操作的,这时候就需要对Runtime的一些基本类型进行了解,我们可以在objc/objc.h中看到以下这些定义

struct objc_object {  
    Class isa  OBJC_ISA_AVAILABILITY;
};

struct objc_class {  
    Class isa  OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
    Class super_class;
    const char *name;
    long version;
    long info;
    long instance_size;
    struct objc_ivar_list *ivars;
    **struct objc_method_list **methodLists**;
    **struct objc_cache *cache**;
    struct objc_protocol_list *protocols;
#endif
};

struct objc_method_list {  
    struct objc_method_list *obsolete;
    int method_count;

#ifdef __LP64__
    int space;
#endif

    /* variable length structure */
    struct objc_method method_list[1];
};

struct objc_method {  
    SEL method_name;
    char *method_types;    /* a string representing argument/return types */
    IMP method_imp;
};

不难发现,基本每个结构都是C语言中的结构体,object_object对应着object,object_class对应着对象所属于的类,我们先把目光主要集中在object_class这个结构体上,可以发现几个上述步骤涉及到的东西:

struct objc_class {  
    Class isa  OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
    Class super_class; // 父类
    const char *name; // 类名
    long version; 
    long info;
    long instance_size;
    struct objc_ivar_list *ivars;
    **struct objc_method_list **methodLists**; // 方法列表
    **struct objc_cache *cache**; // 缓存
    struct objc_protocol_list *protocols;
#endif
};
再次理解消息传递的步骤

理解了Runtime的基本结构后,我们再次用专业的角度来理解消息传递:

  1. 查看method方法是否需要被忽略
  2. 查看object对象是否为nil
  3. 通过objc_object中的isa指针,找到该object的objc_class,然后查看objc_cache类型的cache成员中是否有这个方法,如果有,则找到objc_method中的IMP类型(函数指针)的成员method_imp去找到实现内容,并执行。
  4. 如果没有找到,则查看objc_method_list类型的成员methodLists中是否有该方法
  5. 如果没有找到,则通过Class类型的成员super_class找到父类的objc_class,进行查找
  6. 要是还没有找到,则进入动态解析
如果是调用类方法呢

再次查看这个objc_class的结构:

struct objc_class {  
    Class isa  OBJC_ISA_AVAILABILITY; // metaclass
#if !__OBJC2__
    Class super_class;
    const char *name;
    long version;
    long info;
    long instance_size;
    struct objc_ivar_list *ivars;
    **struct objc_method_list **methodLists**;
    **struct objc_cache *cache**;
    struct objc_protocol_list *protocols;
#endif
};

刚刚第一次看的时候可能没有注意到第一个成员,第一个成员指向的是结构是metaclass,其中包含静态成员变量和静态方法(类方法),同时也包含了一个isa成员,都指向了父类的metaclass,如果是根类,则指向自己。所以如果是调用类方法的话,那么就会利用objc_class中的成员isa找到metaclass,然后寻找方法,没有找到的话则仍然进入动态解析。

动态解析

通过第二次的理解,对于消息传递有了一个清晰的了解,我们继续来研究消息传递最后一步的动态解析。正常我们如果调用了一个没有实现的方法,那么程序会崩溃,并且抛出unrecognized selector to ...的异常,但是利用Runtime,我们可以有三次机会避免程序崩溃,先通过一张图来大致了解下过程:

我们具体看下三种方法:

  • resovleInstanceMethod
void otherEat(id self, SEL cmd) {
    NSLog(@"郑明明");
}

+ (BOOL)resolveInstanceMethod:(SEL)sel {
    if ([NSStringFromSelector(sel) isEqualToString:@"eat"]) {
        class_addMethod(self, sel, (IMP)otherEat, "v@");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

以上需要注意几个地方:

  1. otherEat函数是要被class_addMethod作为参数的,而class_addMethod是Runtime中的API,所以是基于C的,otherEat函数应该是C语言格式的函数
  2. class_addMethod方法可谓是核心,那么依次来看他的参数的含义:
    • first:添加到哪个类
    • second:添加哪个SEL选择器
    • third:IMP函数指针
    • fourth:IMP指针指向的函数返回值和参数类型
      • v@:代表返回值是void类型,无参数
      • i@:代表返回值是int类型,无参数
      • v@:i@:代表返回值是void类型,参数是int类型,存在一个参数(多参数依次累加)
  3. 如果没有调用class_addMethod成功添加方法,那么就会到下一个方法
  • forwardingTargetForSelector
- (id)forwardingTargetForSelector:(SEL)aSelector {
//    return [[Woman alloc]init];
    return nil;
}

如果返回了另外一个对象,那么动态解析又会重新以另外一个对象为接受者执行,如果返回nil,则又继续进入到下一个方法

  • methodSignatureForSelector + forwardInvocation
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    if ([NSStringFromSelector(aSelector) isEqualToString:@"eat"]) {
        return [NSMethodSignature signatureWithObjCTypes:"@v"];
    }
    return [super methodSignatureForSelector:aSelector];
}

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    // 改变消息接受对象
    /*
    Woman *temp = [[Woman alloc]init];
    [anInvocation invokeWithTarget:temp];
     */
    
    // 改变执行的消息
    [anInvocation setSelector:@selector(otherEat)];
    [anInvocation invokeWithTarget:self];
}

methodSignatureForSelector方法返回一个返回值以及参数的封装值,然后会进入到下一个方法,forwardInvocation,这个方法的功能可以说是前两个方法的结合,通过操作NSInvocation对象,既可以改变需执行的消息,又可以改变消息的接受对象

总结

Runtime为Objective-C提供了很多可能,了解消息机制,更加有助于对Objective-C这门语言特性的掌握

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,690评论 0 9
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,548评论 33 466
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,182评论 0 7
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 729评论 0 2
  • 本文详细整理了 Cocoa 的 Runtime 系统的知识,它使得 Objective-C 如虎添翼,具备了灵活的...
    lylaut阅读 795评论 0 4