深入浅出Runtime (二) Runtime的消息机制

已更新
深入浅出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所做的事情

  1. 找到方法的实现。
  2. 调用该方法实现,将接收消息类指针,以及该方法的参数传递给这个类。
  3. 最后将过程的返回值作为自己的返回值传递。

消息传递的关键要素

  1. 指向superclass的指针

  2. 会有一个SEL跟方法实现的地址(这个地址是基于独立的类)关联的表

    当创建一个新的对象时,分配内存,初始化变量,对象变量中的第一个是指向该类结构的指针,这个名字为isa的指针能让对象可以访问它的类,并通过该类访问它继承的所有类

    isa指针是对象使用Objective-C运行时系统所必需的,在结构中定义的任何字段中,对象需要与结构体objc_object(objc/objc.h中的定义)"等效",日常开发中很少有创见自己的根对象的这种情况,一般从NSObject或者NSProxy继承的对象会自动拥有isa变量

Objc-msgSend的传递机制

Objective-C Runtime Programming Guide(官方文档示意图)

Objc-msgSend的发送过程

  1. 消息发送给对象时,消息传递函数遵循对象的isa指针指向类结构的指针,在该结构中它查询结构体变量methodLists中的方法SEL(方法选择器)
  2. 如在isa指向的类结构中找不到SEL(方法选择器),Objc_msgSend会跟随指向Supercalss(父类)指针并再次尝试查找该SEL
  3. 如连续失败直到NSObject类,它的superclass也就是它自己本身
  4. 一旦找到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官方文档

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

推荐阅读更多精彩内容