Objc isa 与class

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

  • class类方法和class 实例方法返回的类相同,
  • superClass类方法和superClass实例方法返回的类相同

这里发现一个问题,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;
}

  • supercClass的类方法和实例方法的实现是相同的.都是返回当前类的superclass成员.
  • class实例方法 和 class类方法的实现不一致, class类方法直接返回当前类, 而class实例方法则调用了object_getClass()运行时函数. 这两个方法中都没有出现isa的身影

object_getClass

Class object_getClass(id obj)
{
    if (obj) return obj->getIsa();
    else return Nil;
}
  • object_getClass 返回了对象的isa,

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

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

结论

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容