Objective-C Runtime(一)对象模型及类与元类

一、什么是Objective-C Runtime

下面是摘自官网的一段原文

The Objective-C language defers as many decisions as it can from compile time and link time to runtime. Whenever possible, it does things dynamically. This means that the language requires not just a compiler, but also a runtime system to execute the compiled code. The runtime system acts as a kind of operating system for the Objective-C language; it’s what makes the language work.

翻译一下:

Objective-C语言会将编译和链接时要做的事推迟到运行时进行,这就意味着Objective-C语言不仅需要一个编译器,还需要一个运行时系统来执行编译好的代码。这个运行时系统就好比Objective-C语言的操作系统, Objective-C基于Objective-C Runtime来工作。

简单的来说,就是Objective-C扩展了 C 语言,并加入了面向对象特性和消息传递机制。而这个扩展的核心是一个用C和汇编语言写的 Runtime 库。它是 Objective-C 面向对象和动态机制的基石。

二、Objective-C 对象模型

对象

Objective-C是一门面向对象的语言,我们无时无刻地在和对象打交道,对象本质上是一个objc_object结构体,他包含一个唯一的私有成员变量isa,可以在objc.h中看到对象的定义:


/// A pointer to an instance of a class.

typedef struct objc_object *id;

/// Represents an instance of a class.
struct objc_object {
    Class isa  OBJC_ISA_AVAILABILITY;
};

对象的isa指针(注:在 64 位 CPU 下,isa 已经不再是一个简单的指针,具体详情可参考Non-pointer isa)指向该对象的类,这个类其实是指向objc_class的结构体,在runtime.h 中可以看到这个结构体的定义:

/// An opaque type that represents an Objective-C class.
typedef struct objc_class *Class;

struct objc_class {
    Class isa  OBJC_ISA_AVAILABILITY; //

#if !__OBJC2__
    Class super_class                                        OBJC2_UNAVAILABLE;//父类
    const char *name                                         OBJC2_UNAVAILABLE;//类名
    long version                                             OBJC2_UNAVAILABLE;
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE;//实例大小
    struct objc_ivar_list *ivars                             OBJC2_UNAVAILABLE;//成员变量列表
    struct objc_method_list **methodLists                    OBJC2_UNAVAILABLE;//方法列表
    struct objc_cache *cache                                 OBJC2_UNAVAILABLE;//方法缓存
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE;//协议列表
#endif

} OBJC2_UNAVAILABLE;
/* Use `Class` instead of `struct objc_class *` */

从结构体的定义可以看到类描述了实例对象的特点,如对象占用的内存大小、成员变量列表、成员函数列表、方法缓存、父类的信息等,关于类的方法列表及方法缓存等,将在下一篇文章中进行介绍。 Class也有一个isa指针,指向一个Class,这个isa指向的Class其实就是类的元类(meta class),同时你可能会有疑问,每一个objc_class都有一个指向objc_class的isa指针,那总有个isa的终点吧?那最终的这个根类的isa指针指向什么呢?

三、类与元类(Meta Class)

上面说到,Class的isa指针指向的是这个类的元类,那什么是元类呢?它的作用又是什么?元类的isa指向的终点是什么?

带着这个问题,我们先来写一个demo:建立两个类SonFather,类Son继承自Father:


@interface Father : NSObject

@end

@interface Son : Father

@end

然后,执行下面这段代码:


Son *son = [Son new];

Class class = object_getClass(son);

do {

    Class isaClass = object_getClass(class);

    id metaClass = objc_getMetaClass(object_getClassName(class));
    Class superClass = class_getSuperclass(class);

    NSMutableString *string = [NSMutableString new];

    [string appendFormat:@"Class %@->isa=%@(%p); superClass=%@(%p); metaclass=%@(%p)\n\n", class, isaClass, isaClass, superClass ?: @"nil", superClass, metaClass, metaClass];

    Class metaClassSuper = class_getSuperclass(metaClass);
    id metaClassIsa = object_getClass(metaClass);

    [string appendFormat:@"metaclass %@(%p)->isa=%@(%p); superClass=%@(%p)\n", metaClass, metaClass, metaClassIsa, metaClassIsa, metaClassSuper, metaClassSuper];

    NSLog(@"%@-------------------------------------\n", string);

    class = superClass;

}while (class);

最终的运行结果是:


2016-02-08 16:11:54.114 ObjcRuntimeDemo[4289:107651] Class Son->isa=Son(0x105d90eb8); superClass=Father(0x105d90f30); metaclass=Son(0x105d90eb8)

metaclass Son(0x105d90eb8)->isa=NSObject(0x1065ec198); superClass=Father(0x105d90f08)
 -------------------------------------
2016-02-08 16:11:54.115 ObjcRuntimeDemo[4289:107651] Class Father->isa=Father(0x105d90f08); superClass=NSObject(0x1065ec170); metaclass=Father(0x105d90f08)

metaclass Father(0x105d90f08)->isa=NSObject(0x1065ec198); superClass=NSObject(0x1065ec198)
 -------------------------------------
2016-02-08 16:11:54.115 ObjcRuntimeDemo[4289:107651] Class NSObject->isa=NSObject(0x1065ec198); superClass=nil(0x0); metaclass=NSObject(0x1065ec198)

metaclass NSObject(0x1065ec198)->isa=NSObject(0x1065ec198); superClass=NSObject(0x1065ec170)
 -------------------------------------

直接看结果可能不太直观,将上述结果绘制成对象模型图:

image

从图中可以看出:一个实例对象的isa指向对象所属的类,这个类的isa指向这个类的元类,而这个元类的isa又指向NSObject的元类,NSObject的元类的isa指向其本身,最终形成形成一个闭环。

在OC中,每一个对象都是类的一个实例,对象的isa指针指向他所属的类,而类本身其实也是一个对象,继承自objc_object,这一点从objc-runtime-new.h中可以看到:


struct objc_class : objc_object {
    // Class ISA;
    Class superclass;
    cache_t cache;             // formerly cache pointer and vtable
    class_data_bits_t bits;    // class_rw_t * plus custom rr/alloc flags
......

当你调用一个类的类方法,如[NSObject alloc]其实是给类对象发送了一个消息, 因为类是一个对象,所以他必须是另外一个类的实例,这个类就是元类,每一个类都有一个isa指针指向这个类所属的类,类的isa指针指向的类就是这个类的元类, 类方法其实是类的元类的实例方法,元类为类对象描述了类方法,就像类为类的实例对象描述实例方法一样, 这一点从runtime的源码objc-class.mm文件中可以看到:

<figure class="highlight" style="box-sizing: border-box; display: block; margin: 0px; background-color: rgb(245, 245, 245); background-position: initial initial; background-repeat: initial initial;">

1
2
3
4
5
6Method class_getClassMethod(Class cls, SEL sel)
{
    if (!cls  ||  !sel) return nil;
    //类的类方法其实是类的元类的实例方法
    return class_getInstanceMethod(cls->getMeta(), sel);
}

</figure>

四、总结

  1. Objective-C中的对象是对C的结构体的封装

  2. 所有的对象都有一个isa指针,指向对象所属的类,类也是一个对象,类对象的isa指针指向类的元类

  3. 类描述了对象的特点,如成员变量、方法列表等,元类描述了类的特点,如类的类方法等

参考链接

本文出自 kingizz's blog - Objective-C Runtime(一)对象模型及类与元类,转载请注明出处。

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

推荐阅读更多精彩内容