OC语言的特性(一)-消息传递与调用函数的表现形式

时间:2021-10-14 14:50:20

我们在初学Objective-C时,都会觉得ObjC中的消息传递和其他语言的调用函数差不多,只是在OC中,方法调用用消息传递这一概念来代替。

那么到底怎样区别OC中的消息传递与其他语言的调用函数呢。

可以使用C语言与OC语言进行一下对比。

C语言示例

//声明一个函数,用来获取两个整型值的中较大的值
int max(int a,int b); int main(int argc, const char * argv[])
{ //调用函数,获取4,6两个整型值的最大值
int result = max(,); //打印结果
printf("4与6两个数的较大的值是:%d",result);
return ;
}

编译运行程序之后,编译器会提示如下错误

Undefined symbols for architecture x86_64:
"_max", referenced from:
_main in main.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

在如上的例子中,我们能够体会到调用函数的语言在声明完函数后,如果没有实现函数,程序是无法编译通过的。

那么我们接下来看

OC的例子

#import <Foundation/Foundation.h>

@interface Function : NSObject

//声明一个类方法,用来获取两个整型值的最大值
+ (int)max:(int)v1 :(int)v2; @end #import "Function.h" @implementation Function @end

只使用Xcode工具的build功能(Command+B),我们能够看到程序是可以编译通过的,但是会有一个黄色的警告

Method definition for 'max::' not found!

只有当程序运行之后才会出现如下的崩溃信息

+[Function max::]: unrecognized selector sent to class 0x1000010f8

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '+[Function max::]: unrecognized selector sent to class 0x1000010f8'
*** First throw call stack:
(
CoreFoundation 0x00007fff8ea9925c __exceptionPreprocess +
libobjc.A.dylib 0x00007fff88555e75 objc_exception_throw +
CoreFoundation 0x00007fff8ea9c02d +[NSObject(NSObject) doesNotRecognizeSelector:] +
CoreFoundation 0x00007fff8e9f7272 ___forwarding___ +
CoreFoundation 0x00007fff8e9f6df8 _CF_forwarding_prep_0 +
SendMethod 0x0000000100000f24 main +
libdyld.dylib 0x00007fff8a8f95fd start +
)
libc++abi.dylib: terminating with uncaught exception of type NSException

通过这种形式的对比,相信大家应该对消息传递和调用函数的形式上的区别应该能够看出来了。

对,消息传递和调用函数对于程序员来说最大的区别就在于源代码编译的过程中是否能够编译通过.

解释消息传递机制的原理就要用到OC语言中的运行时系统(Runtime)了.

运行时系统是一个提供一系列公开函数接口以及数据结构的动态链接库,这些头文件位于/usr/include/objc。许多这些函数允许你使用纯C语言重写当你写OC代码后编译器做的事情。其他形式的接口则是通过NSObject类中定义的一些方法。这些方法是可以用来实现其他的运行时接口来提高运行效率。但是重写运行时的代码对于使用OC语言进行编程并非是必须的,但是,少数的运行时函数在一些特殊情况下,对于OC程序还是很有用途的。

接下来,我们一起看一下这些函数。

  Objc_msgSend()函数举例

  

#import <Foundation/Foundation.h>

@interface Function : NSObject

//声明一个类方法,用来获取两个整型值的最大值
+ (int)max:(int)v1 :(int)v2; @end #import "Function.h" @implementation Function + (int)max:(int)v1 :(int)v2
{
return v1 > v2 ? v1:v2;
} @end

根据以上的Function类,使用objc_msgSend()这种c语言的函数完成oc方法的调用。

    //获取4,6两个整型值的最大值
int result = (int)objc_msgSend([Function class], @selector(max::),,); //打印结果
NSLog(@"4,6的最大值是:%d",result);

消息传递函数为动态绑定提供所有必要的内容:

  • 首先,它找到选择器调用的过程(方法实现)。由于同一个方法在不同的类中可能有不同的实现,这个精确的调用过程依赖于接收者所属于的类。
  • 然后,它会调用这个过程,传递接收者对象(一个指向其数据的指针),以及消息中定义的那些参数。
  • 最后,它传递过程调用的返回值作为它自身的返回值。
  • 注意:编译器会自动调用消息传递函数。你不应该在自己的代码中直接调用该方法。