类结构分析
回顾
前面我们讲了alloc 流程
中对象的创建过程,下面我们来探索一下类的结构,废话不多说,开始~
类的概念 - 本质
什么是类?我们可以在objc源码中找寻class的源头,我们发现Class
对象接受者是objc_class
typedef struct objc_class Class;
然后查看objc_class
的类型是
**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*
class_rw_t *data() {
**return** bits.data();
}
**void** setData(class_rw_t *newData) {
bits.setData(newData);
}
….
}
不难看出类Class
继承于NSObject;在C中编译成Class -> objc_class, NSObject ->objc_object .
C | OC |
---|---|
objc_class | Class |
objc_object | NSObject |
结论:类的本质是objc_object
类型的结构体。
衍生: 要是对其编译后的调用关系有兴趣,可以用终端编译成CPP文件来查看
在调试objc源码中准备以下代码
@interface WXPerson : NSObject@end@implementation WXPerson@end
int main(int argc, const char * argv[]) { @autoreleasepool { WXPerson *p = [WXPerson alloc]; Class cls = object_getClass(p); } return 0;}
利用clang将OC文件输出cpp文件, 指令为:
clang -x objective-c -rewrite-objc -isysroot /Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator.sdk main.m
int main(int argc, const char * argv[]) {
/* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool;
WXPerson *p = ((WXPerson *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("WXPerson"), sel_registerName("alloc"));
Class cls = object_getClass(p);
}
return 0;
}
可以清晰观察出调用的方法,有没很过瘾啊,嘻嘻嘻~
类和元类
类的父类就是元类。比如自定义UI框架里面UIView的TestView,其元类就是UIView,根类是NSObject。以下官方关系图更清晰👇
有个疑惑,类和元类创建时机是什么时候?
通过LLDB指令打印类指针和元类指针
可以看出在main
前打印出类和元类指针,类是在编译时创建了
结论:类和元类的创建时机是在编译期.
类的结构
objc_object是继承于objc_class,所以objc_class也有isa结构体,关系如上图
总结
1.class 是由 superClass,isa,chache_t,bit 组成的结构体
2.通过(class)(isa.bits & ISA_MASK) 可知isa可以强转为class
3.class_rw_t 是在运行时来拓展类的属性,方法和协议等内容
4.实例方法存放在类中,类方法存放在元类里