已更新
深入浅出Runtime (一) 什么是Runtime? 定义?
深入浅出Runtime (二) Runtime的消息机制
深入浅出Runtime (三) Runtime的消息转发
深入浅出Runtime (四) Runtime的实际应用之一,字典转模型
Runtime消息机制
消息发送
在Object-C中,我们其实可以直接调用C的代码也就是Runtime的C语言代码,需要添加message.h
头文件
#import <objc/runtime.h>
#import <objc/message.h>
编写Runtime的时候会遇到没有提示的尴尬,那是因为在Xcode5.0以后的版本,Apple不建议我们写比较底层的代码,So,在target->info搜索msg将YES改成NO,然后可以尽情的使用Runtime代码
其中有个方法objc_msgSend
就是消息发送的方法,在Object-C中所有调用的方法,都是向这个类、或者发送消息
Objc-msgSend所做的事情
- 找到方法的实现。
- 调用该方法实现,将接收消息类指针,以及该方法的参数传递给这个类。
- 最后将过程的返回值作为自己的返回值传递。
消息传递的关键要素
指向
superclass
的指针-
会有一个SEL跟方法实现的地址(这个地址是基于独立的类)关联的表
当创建一个新的对象时,分配内存,初始化变量,对象变量中的第一个是指向该类结构的指针,这个名字为isa的指针能让对象可以访问它的类,并通过该类访问它继承的所有类
isa指针是对象使用Objective-C运行时系统所必需的,在结构中定义的任何字段中,对象需要与结构体objc_object(objc/objc.h中的定义)"等效",日常开发中很少有创见自己的根对象的这种情况,一般从
NSObject
或者NSProxy
继承的对象会自动拥有isa变量
Objective-C Runtime Programming Guide(官方文档示意图)
Objc-msgSend的发送过程
- 消息发送给对象时,消息传递函数遵循对象的isa指针指向类结构的指针,在该结构中它查询结构体变量
methodLists
中的方法SEL
(方法选择器) - 如在isa指向的类结构中找不到
SEL
(方法选择器),Objc_msgSend会跟随指向Supercalss
(父类)指针并再次尝试查找该SEL
- 如连续失败直到
NSObject
类,它的superclass
也就是它自己本身 - 一旦找到
SEL
,该函数就会调用methodLists
的方法并将接收对象的指针传给它
上述过程就是Runtime的是实现方式,在面向对象的编程属于中,方法动态的绑定到消息。
加速消息发送
- 有的时候在一个类会有继承关系,Objective-C中大部分对象都是继承于
NSObject
、自己自定义类,在这种继承体系当中有很多的方法,这些方法有可能不会用到,在向类发送消息的时候,去methodLists
中查找无疑会拖慢程序的运行速度,所以Apple在开发的时候加入了cache
的概念,也就是缓存 - 在每个类中都会有一个单独的缓存,它可以包含继承过来的方法
SEL
以及自己定义的SEL
,在搜索methodLists
之前,消息传递程序会检查接受者对象的告诉缓存cache
,如果找到,就不会在去搜索庞大的methodLists
列表,一旦在缓存当中存在你需要的SEL
,函数调用速度肯定是快的; - 理论上
cache
缓存的是一些会再次调用的SEL
,当写的程序预热足够时间,那么所有发送过的SEL
都会在cache
中找到 -
cache
会动态增长,容纳新的消息,知道程序中所有调用的SEL
运行一遍为止 - 原理是:好比是 通常小圈子找人总比大圈子找人要快
Runtime的发送消息隐藏的参数
每次当我们向一个对象发送消息时,也就是Objective-C调用方法的时候,传递的所有参数,还包括两个隐藏的参数:
- 接收者对象
- 调用的方法
SEL
_cmd
这两个参数没有在定义中声明,而是在编译代码时插入方法实现的。
/*
* _cmd 就是你调用的方法的SEL
**/
NSLog(@"%@",NSStringFromSelector(_cmd));
规避动态绑定的方法,获取方法地址
代码正常编译的时候,需要使用消息传递Objc-msgSend
才能找到方法的IMP
中间就有了这个消息传递的过程
有时候我们不希望调用消息传递的,或者节省消息传递的开销,就需要我们拿到方法的IMP
,代码直接使用IMP
中的方法
下面的示例显示了如何调用实现setFilled:方法的过程:
@interface ViewController ()
{
NSInteger num;
}
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
void (*setter)(id, SEL, BOOL);
int i;
setter = (void (*)(id, SEL, BOOL))[self methodForSelector:@selector(setFilled:)];
for ( i = 0 ; i < 1000 ; i++ )
setter(self, @selector(setFilled:), YES);
}
- (void)setFilled:(NSInteger)number{
NSLog(@"%ld",++num);
}
传递给方法实现的前两个参数是接收对象(self)和方法选择器对象(SEL),这些参数隐藏在方法的语法中,方法作为函数调用时必须使它显式化
使用methodForSelector
绕过动态绑定可以节省消息传递的大部分时间,在特定的消息多次重复的情况下才会节省的更加显著
methodForSelector
是由Cocoa运行时系统提供,它并不是Objective-C语言本身的一个特性
总结
消息机制就是向接收者发送消息,并带有参数,根据接收者对象的数据结构,找到相关发放实现,最后达到这个消息的目的,
objc_msgSend
是Runtime的核心,Objective-C中调用对象方法就是消息传递。
objc_msgSend
并不是直接调用方法实现(IMP)而是发送消息,让类的结构体去动态查到方法实现,所以在为查找到方法实现之前我们可以动态的去修改这个方法的实现
整理转自
apple官方文档