假如调用对象方法的时候,首先从对象的方法列表中查询方法,如果不存在,通过superclass查找父类的方法列表,直到找到方法。
按照这样的逻辑执行的话,每次调用方法都会去查询方法,这样会造成资源的浪费
加入存在一个方法的缓存列表,第一次调用方法的时候,都将这个方法缓存起来,那么下次再调用这个方法的时候,就可以直接从缓存列表中读取,不需要再按照流程去查询方法
再次调用使用过的方法的时候,直接从缓存列表中读取
通过源码分析发现,苹果不是单纯的通过遍历的方式查询方法的位置。 而是通过将 sel 与 mask进行 &预算计算出 方法存放的下标,如果下标内已经存放了方法,会继续往上一个位置存放
如果缓存列表满了,会清空缓存列表,扩容内存为原来的两倍,等待下次调用方法的时候重新进行缓存,因为 _mask 存放的数量 已经改变了,进行 &运算的时候,肯定是和之前的值不一样,所以需要清空,重新缓存
我们OC类在编译后会变成一个objc_class的结构体
objc_class里面包含superclass、cache、bits三部分 其实还与一个隐藏的属性isa(继承objc_object得到)
objc_class结构体继承了objc_object结构体
用一幅图来表示 类的结构大致如下
类其实也是一种对象 我们称之为类对象 类对象的类是元类 这里的isa指向的就是元类
顾名思义就是当前类的父类
这里缓存的是已经调用过的方法 方便再次调用时能快速查找
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方法
2分类中有和主类相同的方法 实际哪个会调用?
答案:分类会调用 因为分类中的方法插入class_rw_t中是向前插入 class_rw_t的方法列表中分类方法在前 主类在后
self = [super init];
这里调用父类的构造函数
有个规则这里是先将父类实例化,才能实例化子类
如果父类实例化失败了 self = nil ,也就无法初始化当前类,父类都初始化不出来,你这个继承于父类的类又还能正常吗?
以上就是关于OC对象的方法缓存全部的内容,包括:OC对象的方法缓存、OC 类的结构分析、IOS开发 中的oc语言方括号中调用[对象\类 方法]怎么理解求解!等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)