360网站建设,windowxp做网站服务器,wordpress唯美主题,做图片的软件app文章目录 category底层结构runtime 执行 category 底层原理添加成员变量 load调用形式系统调用形式的内部原理源码实现逻辑 initialize调用形式源码核心函数#xff08;由上到下依次调用#xff09;如果分类实现了 initialize category
底层结构
本质是结构体。struct _cat… 文章目录 category底层结构runtime 执行 category 底层原理添加成员变量 load调用形式系统调用形式的内部原理源码实现逻辑 initialize调用形式源码核心函数由上到下依次调用如果分类实现了 initialize category
底层结构
本质是结构体。struct _category_t {const char *name;struct _class_t *cls;const struct _method_list_t *instance_methods;const struct _method_list_t *class_methods;const struct _protocol_list_t * protocols;const struct _prop_list_ *properties;
}其中cls 指针的结构为 runtime 执行 category 底层原理
方法名相同时category并不会覆盖 class或者 meta-class 中相同名称的方法实现 消息机制寻找到第一个方法实现则不继续向下寻找在运行时通过runtime动态将分类的方法合并到类对象元类对象中 for (i 0; i used(); i ) 根据分类的方法、属性、协议占用内存大小分别扩充类的 方法列表mlists、 属性列表proplists、 协议列表protolists 每一种列表都是二维数组每一个分类相关数据存储在大数组中的小数组通过 memmove整体移动并覆盖内部会判断移动方向移动类对象的方法、属性、协议到最后通过 memcoy单个移动并覆盖将分类的方法、属性、协议到类中 加载顺序 类优先于分类加载源码采用递归方式保证类加载的优先级 分类之间、类与类之间先编译的先加载后编译先调用
添加成员变量
不能直接添加成员变量但能通过runtime间接添加。property在category中只生成setter和getter方法声明。
方案一 在load方法中完成全局字典初始化对属性进行存取要维护key的唯一性且有线程安全问题内存问题(销毁后仍调用)方案二runtime#import objc/runtime.h在setter方法中调用函数objc_setAssociatedObject(self, key , name, objc_ASSOCIATION_ASSIGN)在getter方法中调用函数return objc_getAssociatedObject(self, key)声明key 全局 staitic const void *key key; 全局 staitic const char key key; // char 减小key内存占用 一定要给key赋初值保证key的唯一性这里是把全局变量key的地址值给了keystatic 保证全局变量只可在文件内访问不使用static在外界可使用extern 读写 直接把key替换为常量字符串(直接声明的字符串放在常量区内存地址不变)直接把key替换为selector(key). 返回的结构体的指针不变
load
调用形式
一个类的 load方法在启动时都会且仅被调用一次重写load系统调用 —— 指针访问直接调用[Class load]手动调用 —— 消息机制
系统调用形式的内部原理
按照编译顺序谁在前面就先被编译先调用完所有类的load方法再调用category的load方法
源码实现逻辑
通过while循环判断是否所有类的load方法都被调用通过递归处理先调用父类load再调用子类load分类通过for 循环取出load_method调用通过do while循环完成所有load方法的调用
initialize
调用形式
消息机制调用 tips: objc_msgSend() —— 该函数底层是使用汇编实现的
调用时机类第一次接收到消息时调用非启动时调用。子类的initialize调用之前先主动调用父类的initialize再调用子类的initialize。initialize 方法是以懒加载的方式被调用的。
源码核心函数由上到下依次调用
实例方法class_getInstanceMethod静态方法class_getClassMethod 内部调用class_getInstanceMethodif (initialize !cls-isInitialized) { 递归 _class_initialize(父类) }
如果分类实现了 initialize
覆盖类本身的initialize调用只执行编译顺序最后那个分类的 initialize