Objective-C之消息机制

话说2014年4月编程语言排行榜中Objective-C的使用比又增加了,看来IOS和MAX OS的开发者是真给力呀。

不过个人感觉用不了多久,IOS和Android的开发者收入就不会有那么大的差异了,因为从现在Android软件的质量上来讲已经有了质的提升。

(扯远了,以上观点纯属虚构,如有雷同,纯属巧合)

 

闲话少说,今儿个聊聊Objective-C的动态绑定。

“消息结构”(messaging structure)

说到动态绑定,就得先说一下Objective-C的“消息结构”(messaging structure)。

"消息结构"与“函数调用”(function calling)还是有很大不同的。

其关键不同在于:

  • 使用消息结构的语言,其运行时所应执行的代码由运行环境来决定。      
  • 使用函数调用的语言,则由编译器决定。

再通俗点说就是:

  采用消息结构的语言,不论是否多态,总是在运行时才会去查找所要执行的方法。

   实际上,编译器设置不关心接收消息的对象是否是何种类型。接收消息的对象问题也要在运行时处理,其过程叫做“动态绑定”(dynamic binding)。

 “动态绑定”(dynamic binding)

在解释动态绑定之前,首先看两段小代码:

 代码段1:

 1 #import <stdio.h>
 2 
 3 void A () {
 4     printf("A");
 5 }
 6 
 7 void B () {
 8     printf("B");
 9 }
10 
11 void callMethod (int value) {
12     if (value == 0) {
13         A ();
14     } else {
15         B ();
16     }
17 }

 

代码段2:

#import <stdio.h>

void A () {
    printf("A");
}

void B () {
    printf("B");
}

void callMethod (int value) {
    void (*func)();
    if (value == 0) {
        func = A
    } else {
        func = B;
    }
    func();
}

下边对以上的代码段进行一下解释:

从功能上来讲,两段代码是一样的,不多说(如果连功能都看不懂的话...咳咳...)

主要是聊一下这两者的区别:

  • 第一段代码是所谓的“静态绑定”(static binding),也就是说,以上代码在程序的编译阶段就能决定运行时所应该调用的函数。

   此时编译器在编译代码的时候就已经知道程序中有A和B这两个函数,并且将这两个函数的地址硬编码子指令之中。

  • 第二段代码是所谓的“动态绑定”(dynamic binding),也就是说,要运行的函数,只有在运行阶段才能决定调用的是A函数还是B函数。

   此时编译器只能等到运行时才会把调用的函数地址硬编码在指令中。

 

如何实现动态调用

 拿这段代码来举例:

  id returnValue = [object messageName:parameter];

如果用“函数调用”的思想来读这段代码的话就是:

  调用object对象的messageName方法,并且传递参数parameter。调用此方法之后返回值为returnValue。

但是!!!Objective-C是“消息结构”!!!

所以正确的理解方式应该是这样:

  • returnValue是返回值,这一点与“函数调用”并没有什么区别。
  • object是消息的“接收者”(receiver),也就是我们给object对象发送了一个消息。
  • messageName,是“选择子”(selector),注意messageName并不是message。那位童鞋问那message是哪个?别着急,咱后边会说。
  • parameter是参数,与“函数调用”没有什么区别。

前边说messageName并不是我们所理解的message,为什么这么说呢?哦~~~?

这么说吧,如果我告诉你去北京找一个叫张三的人,那你肯定就疯了。

但是,如果我告诉你去北京**区**街**小区**栋**室的张铁锤家的张三,是不是很容易呢?

在“消息机制”中,messageName就相当于张三,而object就相当于balabala家的张铁锤。

所以说,“消息”(message)并不是“选择子”(selector),而是“选择子”(selector)与“接收者”(receiver)的组合。

 

怎样传递消息

当编译器收到"id returnValue = [object messageName:parameter]"这条消息的时候,就会将其转换为C语言函数调用(因为Objective-C的底层调用的是C语言),调用的这个函数就是消息传递机制中的核心函数,叫objc_msgSend。

objc_msgSend函数的原型为:void objc_msgSend(id self, SEL cmd, ...)

  • id self 代表消息的接收者
  • SEL cmd 代表选择子
  • ... 代表参数,同时也说明了这是个参数个数可变的函数。

如果将"id returnValue = [object messageName:parameter]"转换为C语言代码的话就应该这么写:

id returnValue = objc_megSend(object, @selector(messageName:), parameter);

objc_megSend函数要做的就是根据接收者与选择子的类型来调用适当的方法。

具体怎么调用的,怎么提高调用的效率,balabala,还是大家有时间自己研究一下吧。

另外附上咱家的微信公众号的二维码,没事儿可以扫扫哈。

有啥事儿大家可以随时留言交流。

posted @ 2014-04-13 23:04  studylifezy  阅读(329)  评论(0编辑  收藏  举报