类本质上也是一个结构体
我们OC类在编译后会变成一个objc_class的结构体
objc_class里面包含superclass、cache、bits三部分 其实还与一个隐藏的属性isa(继承objc_object得到)
objc_class结构体继承了objc_object结构体
用一幅图来表示 类的结构大致如下
类结构体成员分析
1.isa:
类其实也是一种对象 我们称之为类对象 类对象的类是元类 这里的isa指向的就是元类
2.superclass:
顾名思义就是当前类的父类
3.cache:
这里缓存的是已经调用过的方法 方便再次调用时能快速查找
4.bits(重点):
bits调用data()函数 会返回一个class_rw_t的结构体数据
class_rw_t中存放有方法列表、属性列表、协议列表 同时也还有一个class_ro_t
class_ro_t中数据也有方法列表、属性列表、协议列表 但是还多了一个成员变量列表
class_ro_t中是类的本身的原始数据 在编译时就确定了 class_rw_t是运行时生成的 在类的加载过程中class_rw_t初始化并拷贝了class_ro_t中方法列表、属性列表、协议列表等 同时把class_ro_t也保存自己里边 注意 这里并没有将成员变量列表(ivar_list_t)拷贝过去 那么为什么要用到这个class_rw_t呢 这里我们有必要介绍一下分类
分类编译后也同样是一个结构体 如下图 里边同样也有方法列表、协议列表、和属性列表 但是区分了对象方法和类方法列表 类中为什么没有区分呢 因为类中只有对象方法 类方法是类对象的方法 在元类中
在加载分类过程中 会将分类中的方法列表、协议列表、属性列表等插入到相应的class_rw_t中 这里需要注意一点 分类中的方法是向前插入的 也就是说在方法列表中分类的方法会在主类上边
现在想一个问题 我们的方法调用到底查询的哪个方法列表呢 没错 就是class_rw_t里面的方法列表 它既包含了主类的又包含了分类的 这也就是解释了上面的问题 为什么用到class_rw_t
关于这块还有两个面试题分享给大家
1.分类能不能直接使用属性 为什么?
答案:不能 直接使用会产生崩溃 因为找不到setter或getter方法 属性是会默认生成一个成员变量和setter、getter方法 在分类中可以添加属性 但是不会生成成员变量和setter、getter方法 因为成员变量是存放在class_ro_t中的 分类中不能添加成员变量 如果想使用可以通过关联对象的方式自己实现setter和getter方法
@interface Person (category)
@property (nonatomic, strong) NSString *nickName;
@end
#import "Person+category.h"
#import <objc/runtime.h>
@implementation Person (category)
-(void)setNickName:(NSString *)nickName{
objc_setAssociatedObject(self, @"nickName",nickName, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
-(NSString *)nickName{
return objc_getAssociatedObject(self, @"nickName");
}
2.分类中有和主类相同的方法 实际哪个会调用?
答案:分类会调用 因为分类中的方法插入class_rw_t中是向前插入 class_rw_t的方法列表中分类方法在前 主类在后