Objective-C 中的元类(meta class)是什么?

在本文我们会看到一个在Objective-C中很陌生的概念——元类。Objective-C中的每个类都有和自己相关联的元类,但我们几乎从来不直接使用它,它们依然是那么神秘。我们将开始学习怎样在运行时创建一个类。通过创建的“class pair”,我会解释什么是元类,然后探讨它对于Objective-C中对象和类的意义。

在运行时创建一个类

下面的代码在运行时创建了一个NSError的子类,并且添加了一个方法:

Class newClass = objc_allocateClassPair([NSErrorclass],"RuntimeErrorSubclass",0);

class_addMethod(newClass,@selector(report),(IMP)ReportFunction,"v@:");

objc_registerClassPair(newClass);


ReportFunction函数就是添加的实例方法,具体实现如下:


voidReportFunction(idself,SEL_cmd)

{

NSLog(@"This object is %p.",self);

NSLog(@"Class is %@, and super is %@.",[selfclass],[selfsuperclass]);

ClasscurrentClass=[selfclass];

for(inti=1;i<5;i++)

{

NSLog(@"Following the isa pointer %d times gives %p",i,currentClass);

currentClass=object_getClass(currentClass);

}

NSLog(@"NSObject's class is %p",[NSObjectclass]);

NSLog(@"NSObject's meta class is %p",object_getClass([NSObjectclass]));

}



表面上看来,这相当简单。在运行时创建一个类只需要3个步骤:o

为”class pair”分配内存 (使用objc_allocateClassPair).

添加方法或成员变量到有需要的类里 (我已经使用class_addMethod添加了一个方法).

注册类以便它能使用 (使用objc_registerClassPair).

然而,有一个很迫切的问题:什么是“class pair”?objc_allocateClassPair函数仅返回了一个值:the class。那另一半pair在哪?

我相信你已经猜到了,另一半pair就是元类(这篇文章的主题)。为了解释它是什么和我们为什么需要它,还需要交代下Objective-C的对象和类的相关背景。

什么数据结构才能称之为对象?

每个对象都有类。这是面向对象的基本概念,但是在Objective-C中,它对数据结构也一样。含有一个指针且该指针可以正确指向类的数据结构,都可以被视作为对象。

在Objective-C中,对象的类是isa指针决定的。isa指针指向对象所属的类。

实际上,Objective-C中对象最基本的定义是这样的:


typedefstructobjc_object{

Classisa;

}*id;

这说的是:任何带有以指针开始并指向类结构的结构都可以被视作objc_object。

Objective-C中对象最重要的特点是你可以发送消息给它们:

[@"stringValue" writeToFile:@"/file.txt"atomically:YESencoding:NSUTF8StringEncodingerror:NULL];

这能工作是因为Objective-C对象(这儿是NSCFString)在发送消息时,运行时库会追寻着对象的isa指针得到了对象所属的类(这儿是NSCFString类)。这个类包含了能应用于这个类的所有实例方法和指向超类的指针以便可以找到父类的实例方法。运行时库检查这个类和其超类的方法列表,找到一个匹配这条消息的方法(在上面的代码里,是NSString类的writeToFile:atomically:encoding:error方法)。运行时库基于那个方法调用函数(IMP)。

重点就是类要定义这个你发送给对象的消息。

什么是元类

现在,可能你已经知道了,Objective-C的一个类也是一个对象。这意味着你可以发送消息给一个类。

NSStringEncoding defaultStringEncoding=[NSString defaultStringEncoding];

在这个示例里,defaultStringEncoding被发送给了NSString类。

因为Objective-C中每个类本身也是一个对象。如上面所展示的,这意味着类结构必须以一个isa指针开始,从而可以和objc_object在二进制层面兼容,然后这个结构的下一字段必须是一个指向超类的指针(对于基类则为nil)。

正如我上周展示的,类被定义的方式有点不同,依赖于你的运行时库版本,但是,它们都以isa字段开始,随后是superclass字段。


typedefstructobjc_class*Class;

structobjc_class{

Classisa;

Classsuper_class;

/* followed by runtime specific details... */

};

为了调用类里的方法,类的isa指针必须指向包含这些类方法的类结构体。

这就引出了元类的定义:元类是类对象的类。

简单说就是:

当你给对象发送消息时,消息是在寻找这个对象的类的方法列表。

当你给类发消息时,消息是在寻找这个类的元类的方法列表。

元类是必不可少的,因为它存储了类的类方法。每个类都必须有独一无二的元类,因为每个类都有独一无二的类方法。

元类的类是什么?

元类,就像之前的类一样,它也是一个对象。你也可以调用它的方法。自然的,这就意味着他必须也有一个类。

所有的元类都使用根元类(继承体系中处于顶端的类的元类)作为他们的类。这就意味着所有NSObject的子类(大多数类)的元类都会以NSObject的元类作为他们的类

根据这个规则,所有的元类使用根元类作为他们的类,根元类的元类则就是它自己。也就是说基类的元类的isa指针指向他自己。

类和元类的继承

类用super_class指针指向了超类,同样的,元类用super_class指向类的super_class的元类。

说的更拗口一点就是,根元类把它自己的基类设置成了super_class。

在这样的继承体系下,所有实例、类以及元类(meta class)都继承自一个基类。

这意味着对于继承于NSObject的所有实例、类和元类,他们可以使用NSObject的所有实例方法,类和元类可以使用NSObject的所有类方法

这些文字看起来莫名其妙难以理解。Greg Parker给出了一份精彩的图谱来展示这些关系:

实验证明

为了验证,让我们看看我在文章开始写的ReportFunction函数的输出。这个函数的目的是跟随isa指针并打印出它的路途。

为了运行ReportFunction,我们需要创建一个动态实例来创建类调用report方法。

id instanceOfNewClass=

[[newClass alloc]initWithDomain:@"someDomain"code:0userInfo:nil];

[instanceOfNewClass performSelector:@selector(report)];

[instanceOfNewClass release];


这里没有声明report方法,但我使用performSelector:调用它,所以编译器不会给出警告。

然后ReportFunction函数会沿着isa进行检索,来告诉我们class,meta-class以及meta-class的class是什么样的情况:

得到对象的类:ReportFunction函数使用object_getClass跟踪isa指针,因为isa指针是类的保护成员(你不能直接接收其他对象的isa指针)。ReportFunction不使用类方法,因为在类对象里调用类方法不能返回元类,它会再次返回这个类(因此[NSString class]会返回NSString类而不是NSString元类)

This is the output (minusNSLogprefixes) when the program runs:

这是程序运行时的输出(省略了NSlog前缀):


Thisobjectis0x10010c810.

ClassisRuntimeErrorSubclass,andsuperisNSError.

Followingtheisapointer1timesgives0x10010c600

Followingtheisapointer2timesgives0x10010c630

Followingtheisapointer3timesgives0x7fff71038480

Followingtheisapointer4timesgives0x7fff71038480

NSObject's class is 0x7fff710384a8

NSObject'smetaclassis0x7fff71038480

观察isa到达过的地址的值:

对象的地址是0x10010c810.

类的地址是0x10010c600.

元类的地址是0x10010c630.

根元类(NSObject的元类)的地址是0x7fff71038480.

NSObject元类的类是它本身.

这些地址的值并不重要,重要的是它们说明了文中讨论的从类到meta-class到NSObject的meta-class的整个流程。

最后

元类是 Class 对象的类。每个类(Class)都有自己独一无二的元类(每个类都有自己第一无二的方法列表)。这意味着所有的类对象都不同。

元类总是会确保类对象和基类的所有实例和类方法。对于从NSObject继承下来的类,这意味着所有的NSObject实例和protocol方法在所有的类(和meta-class)中都可以使用。

所有的meta-class使用基类的meta-class作为自己的基类,对于顶层基类的meta-class也是一样,只是它指向自己而已。

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

推荐阅读更多精彩内容