iOS 类、元类、Block (基于 OC 2.0)

讲述 objective-c 2.0 对 class 的定义,类、元类的关系,一些面试题,并且对 block 简单说明。

网上很多文章对 class 的定义还是 OC 1.0 的,因此自己写一篇记录一下。

iOS 运行时代码 objc4 的下载地址:
https://opensource.apple.com/tarballs/objc4/

打开源代码,Xcode 同时按 cmd + shift + o,输入 Class 可以搜到相关定义。


1、简化定义

// Class
typedef struct objc_class *Class;
// id
typedef struct objc_object *id;

// 类
struct objc_class : objc_object {
    // 省略部分成员变量以及方法...
}

// 对象
struct objc_object {
    // 省略部分成员变量以及方法...
}

// Object
@interface Object {
    Class isa;
}
@end

// NSObject
@interface NSObject <NSObject> {
    Class isa;
}

可见,类和对象是结构体。

类也是对象,因为对象和类都是结构体 objc_object。


2、详细定义

2.1 对象

结构体 objc_object:

struct objc_object {
private:
    isa_t isa;
    // 省略部分成员变量以及方法...
}

联合体 isa_t:

union isa_t {
    Class cls;
    uintptr_t bits;
    // 省略部分成员变量以及方法...
};

对象有个 isa_t ,isa_t 有个 Class 指向对象的类。
类也是对象,类的 isa_t 的 Class 指向元类。
元类保存类方法,类保存实例方法。

2.2 类

结构体 objc_class:

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

    // class_rw_t 有类的方法和属性列表。
    class_rw_t *data() { 
        return bits.data();
    }
    // 省略部分成员变量以及方法...
}

superclass 是父类,
cache_t 是方法缓存,
class_data_bits_t 包含 class_rw_t,
class_rw_t 里面有类的方法和属性列表。

结构体 class_rw_t:

struct class_rw_t {
    method_array_t methods;
    property_array_t properties;
    protocol_array_t protocols;

    const class_ro_t *ro;
    Class firstSubclass;
    Class nextSiblingClass;

    char *demangledName;
    // 省略部分成员变量以及方法...
};

method_array_t 是个类,保存方法列表,
property_array_t 保存属性列表,
protocol_array_t 保存协议列表。

objc_object 可以简单理解为,有个 isa 指向对象的类,
objc_class 比 objc_object 多了方法列表、属性列表。
然后类、元类也是对象。


3、对象、类、元类的关系

类、元类,也是对象,他们的关系如下图所示:


对象、类、元类关系图

Root class 是 NSObject。

  1. 对于 instance,没有 superclass 线,isa 都指向自己的类。
  2. 对于 class,superclass 指向父类,isa 指向元类。
  3. 对于 meta,superclass 指向父类,isa 指向同一个元类,即 NSObject meta class。
  4. NSObject 没有父类,所以 superclass 指向 nil。
  5. NSObject meta class 的 isa 指向 NSObject 类,形成闭环。

为什么所有元类的 isa 都指向 NSObject meta class?
为什么 NSObject meta class 的 superclass 要指向 NSObject?
有没有大神解答一下😑。

查找对象方法时,根据对象的 isa 找到对象的类,如果子类没有找到方法,就通过 superclass 找到父类,在父类查找方法。

查找类方法时,根据类的 isa 找到元类,如果子元类没有找到,就通过 superclass 找到父元类,在父元类查找方法。

如果根元类,即 NSObject meta class 也没找到,就会去 superclass,也就是 NSObject 类查找。

如果想深入了解,可以看这篇文章:《神经病院Objective-C Runtime入院第一天——isa和Class》。


4、一些题目

4.1

类调用实例方法,是否会崩溃?

// 分类
@interface NSObject (Test)
+ (void)ioo;
@end

@implementation NSObject (Test)
- (void)ioo {
    NSLog(@"%@", NSStringFromSelector(_cmd));
}
@end

测试用例:

- (void)test1 {
    [NSObject ioo];     // 1、
    [NSString ioo];     // 2、
}

1 和 2,哪个会崩溃?

答案是都不会崩溃。

对于 [NSObject ioo] ,先在元类 NSObject meta class 里面找不到 ioo 方法,然后在元类的父类 NSObject class 里找到了。元类 NSObject meta class 的 superclass 指向类 NSObject class 。

对于 [NSString ioo],先在 NSString meta class 里面找不到,然后在 superclass 指向的 NSObject meta class 里也找不到,最后在 NSObject meta class 的 superclass 指向的 NSObject class 里找到了。


4.2

下面的代码是否会崩溃

// 分类
@interface NSString (Test)
+ (void)sioo;
@end

@implementation NSString (Test)
- (void)sioo {
    NSLog(@"%@", NSStringFromSelector(_cmd));
}
@end


- (void)test2 {
    [NSString ioo];    // 1、
    [NSString sioo];    // 2、
}

1 不会崩溃,2 会崩溃。

对于 [NSString sioo],寻找的过程是 NSString meta class、NSObject meta class、NSObject,而方法 sioo 是定义在 NSString 里面,所以崩溃了。

另外,假设只声明但没有实现方法 - (void)fun
对于 [[NSObject new] fun],在 NSObject 里找不到,而 NSObject 的父类是 nil,因此崩溃了。


4.3

下面代码输出什么

- (void)test3 {
    BOOL b1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];
    BOOL b2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]];
    BOOL b3 = [(id)[Son class] isKindOfClass:[Son class]];
    BOOL b4 = [(id)[Son class] isMemberOfClass:[Son class]];
    NSLog(@"%d %d %d %d", b1, b2, b3, b4);

    BOOL b5 = [[Son new] isKindOfClass:[Son class]]; 
    BOOL b6 = [[Son new] isMemberOfClass:[Son class]]; 
    NSLog(@"%d %d", b5, b6); 
    
    BOOL b7 = class_isMetaClass([Son class]); // 要 #import <objc/runtime.h>
    BOOL b8 = [Son new].class == [Son class]; 
    BOOL b9 = class_isMetaClass([[Son class] class]);
    NSLog(@"%d %d %d", b7, b8, b9);
}

b1-4 输出 1 0 0 0,
b5-6 输出 1 1,
b7-9 输出 0 1 0。

先看 class 方法:

+ (Class)class {
    return self;
}

- (Class)class {
    return object_getClass(self);
}

Class object_getClass(id obj)
{
    if (obj) return obj->getIsa();
    else return Nil;
}

方法 object_getClass 是获取对象的 isa 指向的 class。
类和对象的 class 方法都返回类,不是元类。
类的 class 方法返回 self,对象的返回所属类。

举个栗子:

po [Son class]
po [[[[Son class] class] class] class]

上面都是输出 Son,不管调用多少次 class,返回的都是 self

然后看下 isMemberOfClass 方法:

+ (BOOL)isMemberOfClass:(Class)cls {
    return object_getClass((id)self) == cls;
}

- (BOOL)isMemberOfClass:(Class)cls {
    return [self class] == cls;
}

- (Class)class {
    return object_getClass(self);
}

Class object_getClass(id obj)
{
    if (obj) return obj->getIsa();
    else return Nil;
}

方法 object_getClass 是获取对象的 isa 指向的 class,如果传入的参数是类对象,获取的就是元类了。
方法 isMemberOfClass 会取一次 isa 指向的类,然后进行比较。
如果是对象调用 isMemberOfClass,就是类与类比较。
如果是类调用 isMemberOfClass,就是元类与类比较。

举个栗子:

[(id)[A class] isMemberOfClass:[B class]];

上面代码是 A meta class,与 B class 进行比较,
而不是 A class 与 B class 比较。

最后看下 isKindOfClass :

+ (BOOL)isKindOfClass:(Class)cls {
    for (Class tcls = object_getClass((id)self); tcls; tcls = tcls->superclass) {
        if (tcls == cls) return YES;
    }
    return NO;
}

- (BOOL)isKindOfClass:(Class)cls {
    for (Class tcls = [self class]; tcls; tcls = tcls->superclass) {
        if (tcls == cls) return YES;
    }
    return NO;
}

- (Class)class {
    return object_getClass(self);
}

Class object_getClass(id obj)
{
    if (obj) return obj->getIsa();
    else return Nil;
}

可见 isKindOfClass 就是 isMemberOfClass 的循环版,会通过 superclass 沿着继承链进行判断。
如果是对象调用 isMemberOfClass,会沿着 class 继承链判断。
如果是类调用 isMemberOfClass,会沿着 meta class 继承链判断。
需要注意的是,NSObject meta class 的 superclass 是 NSObject class,isa 指向自己。
NSObject class 的 superclass 是 nil,isa 指向 NSObject meta class。

对于
BOOL b1 = [(id)[NSObject class] isKindOfClass:[NSObject class]]
先是 NSObject meta class 与 NSObject class 比较,
然后是 NSObject class 与 NSObject class 比较,所以结果是 1。

总结一下:

- (void)test3 {
    // 类和对象的 class 方法会返回类,不是元类。
    // 类的 class 方法返回 self,对象的返回所属类。
    // isMemberOfClass 会取 isa 指向的类,与参数的类就行比较
    // 如果是对象调用 isMemberOfClass,就是类与类比较
    // 如果是类调用 isMemberOfClass,就是元类与类比较
    // isKindOfClass 会通过 superclass 沿着继承链循环判断
    
    BOOL b1 = [(id)[NSObject class] isKindOfClass:[NSObject class]]; // 1,循环到 NSObject == NSObject
    BOOL b2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]]; // 0,NSObject meta != NSObject
    BOOL b3 = [(id)[Son class] isKindOfClass:[Son class]]; // 0,从 Son meta class 循环到 NSObject 到 nil,都 != Son
    BOOL b4 = [(id)[Son class] isMemberOfClass:[Son class]]; // 0,Son meta != Son
    b3 = [Son isKindOfClass:[Son class]]; // 0,和上面的两句是一样的
    b4 = [Son isMemberOfClass:[Son class]]; // 0,Son meta != Son
    NSLog(@"%d %d %d %d", b1, b2, b3, b4);// 1 0 0 0
    
    BOOL b5 = [[Son new] isKindOfClass:[Son class]]; // 1
    BOOL b6 = [[Son new] isMemberOfClass:[Son class]]; // 1,Son == Son
    NSLog(@"%d %d", b5, b6); // 1 1
    
    
    BOOL b7 = class_isMetaClass([Son class]); // 0,要 #import <objc/runtime.h>
    BOOL b8 = [Son new].class == [Son class]; // 1,Son == Son
    BOOL b9 = class_isMetaClass([[Son class] class]); // 0,[[Son class] class] == Son
    NSLog(@"%d %d %d", b7, b8, b9);// 0 1 0
}



5、Block

最后简单说一下 block。block 也是对象。

定义一个继承自 NSObject 的 Test 类,有个 test 方法:

@implementation Test

- (void)test {
    void (^blc)(void) = ^ {
        printf("哈哈哈");
    };
    
    blc();
}

@end

代码转换后, block 语法转换的函数:

// block 的函数,参数是一个 block 指针
static void __Test__test_block_func_0(struct __Test__test_block_impl_0 *__cself) {
    printf("哈哈哈");
}

block 转换成结构体:

// 自定义 block 的结构体
struct __Test__test_block_impl_0 {
    struct __block_impl impl; // block 的基本定义
    struct __Test__test_block_desc_0* Desc; // block 的数据
    
    // 构造函数。
    // fp 是 block 的函数的指针,desc 是 block 的数据。
    __Test__test_block_impl_0(void *fp, struct __Test__test_block_desc_0 *desc, int flags=0) {
        // 省略部分代码
    }
};

再看 block 的基本定义:

// block 的基本定义
struct __block_impl {
    void *isa; // block 的类
    int Flags;
    int Reserved;
    void *FuncPtr; // 指向 block 的函数
};

里面有个 isa 指针,指向 block 的类,可以是堆、栈、全局 block。

打印输出 block 的继承链:

(lldb) po [blc class]
__NSGlobalBlock__

(lldb) po [blc superclass]
__NSGlobalBlock

(lldb) po [[blc superclass] superclass]
NSBlock

(lldb) po [[[blc superclass] superclass] superclass]
NSObject

(lldb) po [[[[blc superclass] superclass] superclass] superclass]
nil

具体可以看我的另一篇文章 笔记-《Objective-C高级编程 iOS与OS X多线程和内存管理》,在 2.3 章节记录 Blocks 的实现。


如有错误,欢迎指正。

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

推荐阅读更多精彩内容

  • Objective-C语言是一门动态语言,它将很多静态语言在编译和链接时期做的事放到了运行时来处理。这种动态语言的...
    有一种再见叫青春阅读 587评论 0 3
  • iOS底层原理总结 - 探寻OC对象的本质 原文链接 对小码哥底层班视频学习的总结与记录。面试题部分,通过对面试题...
    二斤寂寞阅读 649评论 0 4
  • 六点五十三分,列车到达徐州。 天终于白的发亮。 漫长的黑夜终于看到了尽头,开往上海的列车上,大多数人都在酣睡,只有...
    灯下尘_b6b2阅读 351评论 0 0
  • 美杜莎从万科城18楼走楼梯一步一步的走上21楼天台,前几天的天气不好,微雨,天台地面上哪里都是湿淋淋的,没有人上来...
    NeedUArya阅读 290评论 0 1
  • 班主任很让人讨厌。90%的班主任都很让人讨厌。让人讨厌的不是班主任,是这个教育环境。让人讨厌的是某些人性。 孩子是...
    梁森的简书阅读 201评论 0 1