初识objc_msgSend

动态绑定(Dynamic Binding)

在面向对象语言中,大家最常干的一件事就是调用一个对象的方法。在OC的术语中,此过程被称为“发送消息”。一条消息,它拥有名字,亦或称之为“选择器”,它接受参数,并且还会有返回值。

因为OC是C的一个超集,可以先理解在C中函数是如何被调用的,在此基础之上会更容易理解OC中的消息发送。先看一个例子:

void printHello() {
  printf("Hello, world!\n");
}
void printGoodbye() {
  printf("Goodbye, world!\n");
}
void doTheThing(int type) {
  if (type == 0) {
    printHello();
  } else {
    printGoodbye();
  }
  return 0;
}

此上这种函数调用的方法被称为“静态绑定”(static binding)。何为“静态”。它表明在程序编译阶段,编译器已经知道哪个函数会被调用。当以上代码被编译的时候,编译器已经知道了printHello函数同printGoodbye函数,并且编译器会发出指令来直接调用这两个函数。这两个函数的地址被硬编码进指令中。

再来看一个例子:

void printHello() {
  printf("Hello, world!\n");
}
void printGoodbye() {
  printf("Goodbye, world!\n");
}
void doTheThing(int type) {
  void (*fnc)();
  if (type == 0) {
    fnc = printHello;
  } else {
    fnc = printGoodbye;
  }
  fnc();
  return 0;
}

此种函数调用的策略,被称为“动态绑定”。何为“动态”(dynamic binding)?只有在程序运行的时候,才能准确地知道哪个函数被调用。以上两段代码在编译阶段生成的指令是不一样的。在第一段代码中,在if和else中都出现了函数调用;而在第二段代码中,函数调用只出现了一次,但是只有在运行阶段,才能得知调用的到底是printHello函数还是printGoodbye函数。

OC中的消息发送

在OC中,消息发送会引起底层的函数调用,采取的策略就是动态绑定。在OC这层光鲜亮丽的建筑之下,所有的方法皆是朴实无华的C函数。至于究竟是哪个C函数会被调用,这完全是在运行时被决定的,这些函数甚至可以在你APP运行的时候被改变。因此,我们称Objective-C是一门完全动态的语言。

一则消息的结构如下:

id returnValue = [someObject messageName:parameter];

someObject被视作消息接收方,messageName是一个选择器。选择器加上参数就被称为“消息”。当编译器看见这则消息的时候,它会将其转化成一个标准的C函数:

void objc_msgSend(id self, SEL cmd, ...)

objc_msgSend函数接收两个或者两个以上的参数,第一个参数是消息的接收方,第二个参数是选择器(SEL是选择器的类型,就像int是一个数的类型一样),剩下的参数视发送的消息的情况而定。一个选择器是一个名字,该名字🈶指明了一个方法。上一句消息发送的代码会被转换成如下C函数:

id returnValue = objc_msgSend(someObject, @selector(messageName:), parameter);

那么objc_msgSend函数怎么样才能调用到正确的方法呢?这就决定于消息的接收方和选择器了。为简单模型,先只考虑消息的接收方是一个类的对象,该类维护了一个列表,在这个列表中存放的是方法的具体实现。为了保证方法的正确调用,objc_msgSend会去查看消息接收方的这个列表,并且去寻找匹配选择器的方法。如果找到了,就跳转到方法的实现代码块;如果没有找到,就沿着继承链向上查找。如果没有找到匹配的方法,就会转如Message Forwarding(这个术语以及它牵扯出的runtime知识笔者会在下篇文章中讨论)。

在这整个过程中,决定哪个方法被调用似乎成本很高,因为看上去有很多的工作要做。但OC的runtime机制是聪明的,objc_msgSend会将结果缓存在一张高效率的哈希表中,每一个类都拥有这样的一张表,因此同一个类在接收到新的消息后,执行效率并不低。但是,这种聪明的方法只是在“慢”的基础上让它不那么“慢”,当然还是没有静态绑定的函数调用快速的。

OC对象的任一方法都可以看作是一个简单的C函数:

<return_type> Class_selector(id self, SEL _cmd, ...)

它真实的样子并不长这样,这样写是为了便于理解。每个类都维护了一个表格,“键”就是选择器,“值”就是一个函数指针。这就是objc_msgSend能正确调用函数的秘诀。

尾优化(Tailing-Call Optimization)

当一个函数执行到最后的时候,最后一条语句是调用另一个函数,且被调用函数的返回值并不对原函数程序产生任何的影响,在此种情况下,栈的工作情况如下:栈顶元素弹出,即原函数的栈空间被回收,被调用函数的栈空间压入栈中。而不是我们认为的那样:原函数的栈空间依然被保存,被调用函数的栈空间压入栈中。

可以假想一下,如果没有这种优化策略,在每次调用一个OC对象的方法之前,栈空间上都会出现objc_msgSend函数的栈空间,同事不要忘记,objc_msgSend还有很多朋友们,它们之间相辅相成,这样的话,会导致栈空间过早地溢出。


译文原作:《Effective Objective-C 2.0》Item11
译作者:WishQi

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,698评论 0 9
  • 消息发送和转发流程可以概括为:消息发送(Messaging)是 Runtime 通过 selector 快速查找 ...
    lylaut阅读 1,829评论 2 3
  • 前言 runtime其实在我们日常开发过程中很少使用到,尤其是像我现在比较初级的程序猿就更用不到了。但是去面试很多...
    WolfTin阅读 620评论 0 2
  • 相遇,是一个多么动人的名词。 在这个世界上,两个人相遇的可能性是千万分之一,成为朋友的可能性是两亿分之一,而成为终...
    甜如密意阅读 272评论 0 1
  • 长这么大除了长辈外头一次有女孩纸主动煮东西给我吃! 谢主隆恩,万岁万岁万万岁!
    顾问的顾阅读 187评论 2 0