Runtime窥探 (二)| 消息发送

前言

双眸浅看,几世繁华褪色成一纸墨色,岁月索颜,在风居住的街道,我怀抱相思,在雨中等待前世,等待今世,等待晴天,亦或是等待你。

作品集

1.例子

  • 新建Person类,继承于NSObject
  • 新建方法:
- (void)eat{
    NSLog(@"实例方法正在吃饭。。。。");
}
- (void)eatWith:(NSString *)name{
    NSLog(@"实例方法正在和%@吃饭。。。。",name);
}
- (void)drink{
    NSLog(@"实例方法正在喝水。。。。");
}
    
+ (void)eat{
    NSLog(@"类方法正在吃饭。。。。");
}

看看下面例子会执行吗,如果执行打印出什么?

- (void)objc_message{
    Person *p = [Person new];
    [p eat];
    objc_msgSend(p,@selector(eatWith:),@"Dely");
    [p performSelector:@selector(eat) withObject:nil];
    [[p class] performSelector:@selector(eat) withObject:nil];
}

结果是肯定可以执行的,打印结果如下:

实例方法正在吃饭。。。。
实例方法正在和Dely吃饭。。。。
实例方法正在吃饭。。。。
类方法正在吃饭。。。。

疑问:

  • objc_msgSend是个什么东西?
  • [[p class] performSelector:@selector(eat) withObject:nil];为什么可以这样调用?

2.objc_msgSend是什么?

objc_msgSend:消息发送、负责方法的调用。

直接使用objc_msgSend需要导入头文件#import <objc/message.h>

//方法定义:消息接受者,消息,参数
objc_msgSend(id _Nullable self, SEL _Nonnull op, ...)
  • 当我们发送消息[receiver message]时,会被编译成:
objc_msgSend(receiver,@selector(message), ...)

  • 也就是说:调用[p eat]时;其实就是调用了
//无参数
objc_msgSend(p,@selector(eat));
//有参数
objc_msgSend(p,@selector(eatWith:),@"Dely");
  • performSelector方法底层其实就是基于objc_msgSend封装的,被编译成
objc_msgSend(Person,@selector(eat));

3.消息发送机制

我们就算调用了objc_msgSend方法那到底怎么调用到函数的实现的呢?他的调用机制是怎么样的呢?也就是我们常说的OC的消息发送机制.

我们在上一篇讲解isa和class的时候无形中其实也讲解了消息发送机制,主要讲解了isa和class,不明白的可以再去复习一下isa和class到底是个什么东西?

objc_msgSend如何执行方法?

上面提到了,一个OC方法被编译成objc_msgSend,那么,Runtime如何找到方法的执行体呢?

再看下runtime中objc_class定义:

//Class
typedef struct objc_class *Class;

//objc_class
struct objc_class {
    Class _Nonnull isa  OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
    Class _Nullable super_class                              OBJC2_UNAVAILABLE;
    const char * _Nonnull name                               OBJC2_UNAVAILABLE;
    long version                                             OBJC2_UNAVAILABLE;
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE;
    struct objc_ivar_list * _Nullable ivars                  OBJC2_UNAVAILABLE;
    struct objc_method_list * _Nullable * _Nullable methodLists                    OBJC2_UNAVAILABLE;
    struct objc_cache * _Nonnull cache                       OBJC2_UNAVAILABLE;
    struct objc_protocol_list * _Nullable protocols          OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

可以看到 Class 中包含了isa指针、methodLists、cache等信息。

  • 实例可以通过isa指针找到Class
  • Class中methodLists包含了类中所有的实例方法
  • 实例的元类中methodLists中包含了类中所有的加方法
objc_message

实例方法的执行过程:

例如:[p eat];

  • 1.首先被编译成objc_msgSend(id receiver, SEL op, ...);
  • 2.根据receiver对象的isa指针获取它对应的Class;
  • 3.优先在class的cache(为了提高效率)查找message方法,找到直接执行执行6,找不到继续往下执行
  • 4.再到methodLists查找,找到直接执行6,找不到继续往下执行
  • 5.没有在class找到,再到super_class查找,依次查找到基类NSObject.如果还没找到执行消息转发机制(本文不讲解)
  • 6.找到method,根据IMP执行函数,并将返回值给调用者

类方法的执行过程:

例如:[Person eat];

因为类方法列表保存在元类中,所以跟实例方法不同的是:首先类对象会根据Class的isa指针找到对应的元类。查找过程在元类中的methodLists查找方法,找不到的话,查找到当前的父类,然后在父类对应元类的methodLists。执行过程就和上面的实例方法一样了。

  • 每一个Class的isa指向唯一一个元类,每个元类的isa都指向同一个根元类
  • 其实类Class本身就是一个实例,只不过他是元类的实例。所以我们根据实例的isa就可以找到对应的类

开始还有一个疑问? [[p class] performSelector:@selector(eat) withObject:nil];为什么可以这样调用?
编译完成后都会执行objc_msgSend,只不过对象是Class类型罢了,给对象发一个消息。依次查找元类中的方法,当然可以执行了。

现在对于OC的消息转发机制是否有一些了解。上面说到消息到最后都没有找到该怎么办?他该何去何从?会立即崩溃吗?这就是下一篇博客要讲的内容:消息转发

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,679评论 0 9
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 727评论 0 2
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,544评论 33 466
  • 在Objective-C中,使用 [receiver message] 语法并不会马上执行receiver对象的...
    Stago阅读 196评论 0 0
  • 前言 runtime其实在我们日常开发过程中很少使用到,尤其是像我现在比较初级的程序猿就更用不到了。但是去面试很多...
    WolfTin阅读 611评论 0 2