02 iOS底层原理 - isa和superclass指针探究

废话不多说,先看两道面试题:

  1. OC对象的isa指针指向哪里?
    instance的isa指向calss对象
    class的isa指向meta-class
    meta-class的isa指向基类的meta-class
  2. OC对象的类型信息存放在哪里?
    对象方法、属性、成员变量、协议信息,存放在class对象中
    类方法,存放在meta-class对象中
    成员变量的具体值,存放在instance对象中

通过这两个面试题,咋们一点点揭开isa和superclass的神秘面纱

一,创建NSObject分类Test、Person、Student类

1. NSObject分类 Test
// 声明
@interface NSObject (Test)
+ (void)test;
@end

// 实现
@implementation NSObject (Test)
+ (void)test {
    NSLog(@"+[NSObject test] - %p", self);
}
- (void)test {
    NSLog(@"-[NSObject test] - %p", self);
}
@end
2. Person类,继承自NSObject
// 声明
@interface Person : NSObject <NSCopying>
{
    @public
    int _age;
}

// 实现
@property (nonatomic, assign) int no;
- (void)personInstanceMethd;
+ (void)persongClassMethod;
+ (void)test;
@end
3. Student类,继承自Person
// 声明
@interface Student : Person<NSCoding>
{
    @public
    int weight;
}
- (void)studentInstanceMethod;
+ (void)studentClassMethod;
@end

// 实现
@implementation Student
- (void)studentInstanceMethod {
}
+ (void)studentClassMethod {
}
- (instancetype)initWithCoder:(NSCoder *)coder {
    return nil;
}
- (void)encodeWithCoder:(NSCoder *)coder {
}

二,开始研究isa和superclass指针的本质

1. 对象方法调用的本质

调用personInstanceMetod这个对象方法,利用runtime消息机制,给person发送消息,具体runtime的内容以后会有章节讲解。

Person *person = [[Person alloc]init];
person->_age = 10;

/*
  调用personInstanceMetod这个对象方法
  利用runtime消息机制,给person发送消息
  问题:实例对象里面并没有这个对象方法,这个方法是存在于类对象里面的,那么到底是怎么发送的呢?
*/
[person personInstanceMethd];
// objc_msgSend(person, @selector(personInstanceMetod))
2. 类方法调用的本质

调用personClassMethod这个类方法,利用runtime消息机制,给Person这个类对象发消息

/*
  调用personClassMethod这个类方法
  利用runtime消息机制,给[Person class] 发消息
  问题:类对象里面并没有这个类方法,这个方法是存在于元类对象里面的,那么到底是怎么发送的呢?
*/
 [Person persongClassMethod];
// objc_msgSend([Person class], @selector(personClassMethod))

那么问题来了:

  1. 实例对象里面并没有这个对象方法,而这个对象方法是存在于类对象里面的,那么到底是怎么发送的呢?
  2. 类对象里面并没有这个类方法,这个方法是存在于元类对象里面的,那么到底是怎么发送的呢?
    贴两张图,大家就会看得明白了:


    image.png

从上面的示意图可以看出:

  1. instance的isa指针指向class,当调用对象方法时(personInstanceMetod),通过instance的isa找到对应的class,最后在class中找到对象方法的实现进行调用。
  2. class的isa指针指向meta-class,当调用类方法时(personClassMethod),通过class的isa找到对应的meta-class,最后在meta-class中找到类方法的实现进行调用
3. class对象的superclass指针
image.png

子类调用父类的对象方法,先通过isa找到子类的class,然后再通过class的superclass找到父类,这样一层层往上找,直到找到对象方法的实现进行调用。

/*
子类调用父类的对象方法,先通过isa找到子类的class,然后在通过class的superclass找到父类
这样一层层往上找,直到找到对象方法的实现进行调用
*/
Student *student = [[Student alloc]init];
// isa可以找到class,在class里面找到对象方法
[student studentInstanceMethod];
// 分析调用流程
/*
当Student的instance对象要调用Person的对象方法时,
先通过isa找到Student的class,
然后通过superclass找到Person的class,最后找到对象方法的实现进行调用
*/
[student personInstanceMethd];
4. meta-class对象的superclass指针
image.png

子类调用父类的类方法,先通过isa找到子类的meta-class,然后再通过meta-class的superclass找到父类,这样一层层往上找,直到找到类方法的实现进行调用

/*
子类调用父类的类方法,先通过isa找到子类的meta-class,然后再通过meta-class的superclass找到父类
这样一层层往上找,直到找到类方法的实现进行调用
*/
// 通过Student类对象的isa找到,Studet对象的meta-class,最后找到类方法的实现进行调用
[Student studentClassMethod];
// 分析调用流程
/*
当Student的class对象要调用Person的类方法时,
先通过isa找到Student的meta-class,
然后通过superclass找到Person的meta-class,最后找到类方法的实现进行调用
*/
[Student persongClassMethod];
5. 总结一下isa 和 superclass指针

其实将上面的三张图总结到一张图赏就很一目了然了:


image.png

这张图标注的还是很清楚的了,但是需要注意一点的是:
当元类(meta-class)通过superclass找到了基类(Root class),这是还没有找到相应的类方法,那么此时基类就会通过superclass找到类(class),在class里面找对应的实例方法,还是找不到superclass就会置为nil。如下图所示的箭头:


image.png

三,分析调用Test方法的调用轨迹

根据上面对isa和superclass的研究,现在分析下下面两句代码的调用轨迹

NSLog(@"[Person class] - %p", [Person class]);
NSLog(@"[NSObject class] - %p", [NSObject class]);
[Person test];
[NSObject test];
  • 情况1. Person中有 +test类方法,NSObject中也有 +test类方法:
    通过Person isa 找到该类的meta-class,然后在meta-class中找相应的类方法,发现有这个类方法,那么就调用执行
    * 情况2. Person中没有 +test类方法,NSObject中也有 +test类方法:
    通过Person isa 找到该类的meta-class,然后在meta-class中找相应的类方法,发现没有这个类方法,
    那么就通过superclass去找父类,在父类NSObject中发现有这个类方法,那么就调用执行
    * 情况3. Person中没有 +test类方法,NSObject中也没有 +test类方法,Person中没有 -test对象方法:
    通过Person isa 找到该类的meta-class,然后在meta-class中找相应的类方法,发现没有这个类方法,
    那么就通过superclass去找父类,在父类NSObject中发现也没有这个类方法,此时还是通过superclass找到class,
    在class中看看有没有 -test 对象方法,发现没有,再往上找,superclass就会置为Nil,
    最后报错 unrecognized selector sent to class 0x1000014c0
    * 情况4. Person中没有 +test类方法,NSObject中也没有 +test类方法,但是NSObject中有 -test对象方法:
    通过Person isa 找到该类的meta-class,然后在meta-class中找相应的类方法,发现没有这个类方法,
    那么就通过superclass去找父类,在父类NSObject中发现也没有这个类方法,此时还是通过superclass找到class,
    在class中看看有没有 -test 对象方法,发现有,那么就调用执行

四,OC对象的类型信息存放在哪呢

OC对象的类型信息存放在哪,如下图所示:


image.png

为了更好的证明确实是这样存放的,可以查看源码
https://opensource.apple.com/tarballs/objc4/

关键源码:

image.png

所以,文章开头所提面试题,OC对象的类型信息存放在哪里?

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

推荐阅读更多精彩内容