Objc isa 与class

2020-02-03  本文已影响0人  好有魔力
image.png

最近在看这张经典图时,突然有点懵逼,对isa的概念有点模糊了,因为在开发中通常使用 class,class 就是读取isa吗?

通过代码测试一下

@interface Person :NSObject

@property(nonatomic,strong) NSString *string;

@end

@implementation Person

@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {

       Person *person = [[Person alloc] init];
        
        NSLog(@"person.class : %@ %@  meta",
              NSStringFromClass(person.class),
              class_isMetaClass(person.class)?@"is":@"isn't");
        
        NSLog(@"person.superClass: %@ %@  meta",
              NSStringFromClass(person.superclass),
              class_isMetaClass(person.superclass)?@"is":@"isn't");
        
        NSLog(@"Person.class : %@ %@  meta",
              NSStringFromClass(Person.class),
              class_isMetaClass(Person.class)?@"is":@"isn't");
        
        NSLog(@"Person.superClass: %@ %@  meta",
              NSStringFromClass(Person.superclass),
              class_isMetaClass(Person.superclass)?@"is":@"isn't");
        
        NSLog(@"break point");
    }
    return 0;
}

打印的结果:
 person.class : Person isn't  meta
 person.superClass: NSObject isn't  meta
 Person.class : Person isn't  meta
 Person.superClass: NSObject isn't  meta

这里发现一个问题,Person.class如果是读取isa的话,其打印的结果应该是元类. 但是测试结果却不是这样

结合objc750源码来看看内部的实现

objc750源码探索

+ (Class)class {
    return self;
}

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

+ (Class)superclass {
    return self->superclass;
}

- (Class)superclass {
    return [self class]->superclass;
}

object_getClass

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

调用class类方法并非获取isa, 只是简单地返回当前类.而调用class实例方法是返回isa.

对于一些情况,class 实例方法并不是总是返回对象的isa.例如,使用KVO监听了对象的某一属性之后,该对象的class方法 和 isa 都会被重写导致调用该对象的class方法 与 object_getClass()返回的结果并不一致. 对外界隐藏实现了细节,并且可以轻松替换对象实际类型.

结论

isa 表示对象的真实类型,无论该对象是类对象还是实例对象,因为系统使用isa来查找方法和属性信息. 而class实例方法返回的类型并不一定"真实",但是对API使用无影响

上一篇 下一篇

猜你喜欢

热点阅读