当前位置: 首页 > news >正文

iOS OC 底层原理之 category、load、initialize

文章目录

    • 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,动态将分类的方法合并到类对象,元类对象中:
    1. for (i = 0; i < used(); i ++)
      根据分类的方法、属性、协议占用内存大小,分别扩充类的:
      方法列表mlists、
      属性列表proplists、
      协议列表protolists
      每一种列表都是二维数组,每一个分类相关数据存储在大数组中的小数组
    2. 通过 memmove(整体移动并覆盖,内部会判断移动方向)移动类对象的方法、属性、协议到最后
    3. 通过 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:
    1. 全局 staitic const void *key = &key;
    2. 全局 staitic const char key = &key; // char 减小key内存占用
      • 一定要给key赋初值,保证key的唯一性
      • 这里是把全局变量key的地址值给了key
      • static 保证全局变量只可在文件内访问
      • 不使用static,在外界可使用extern 读写
    3. 直接把key替换为常量字符串(直接声明的字符串放在常量区,内存地址不变)
    4. 直接把key替换为@selector(key). 返回的结构体的指针不变

load

调用形式

  • 一个类的 load方法在启动时都会且仅被调用一次
  • 重写+load,系统调用 ——> 指针访问直接调用
  • [Class load],手动调用 ——> 消息机制

系统调用形式的内部原理

  • 按照编译顺序,谁在前面就先被编译
  • 先调用完所有类的load方法
  • 再调用category的load方法

源码实现逻辑

  1. 通过while循环,判断是否所有类的load方法都被调用
  2. 通过递归处理,先调用父类+load,再调用子类+load
  3. 分类通过for ++ 循环,取出load_method调用
  4. 通过do while循环,完成所有load方法的调用
    在这里插入图片描述

initialize

调用形式

消息机制调用
tips: objc_msgSend() ——> 该函数底层是使用汇编实现的

  • 调用时机
  • 类第一次接收到消息时调用,非启动时调用。
  • 子类的initialize调用之前,先主动调用父类的initialize,再调用子类的initialize。
  • initialize 方法是以懒加载的方式被调用的。

源码核心函数(由上到下依次调用)

  1. 实例方法:class_getInstanceMethod
  2. 静态方法:class_getClassMethod (内部调用class_getInstanceMethod)
  3. if (initialize && !cls->isInitialized) { 递归 _class_initialize(父类) }

如果分类实现了 +initialize

  • 覆盖类本身的+initialize调用
  • 只执行编译顺序最后那个分类的 + initialize

相关文章:

  • 通过 LabVIEW 正则表达式读取数值(整数或小数)
  • 软件设计之SSM(3)
  • 从《GTA5》的反外挂斗争看网络安全的重要性
  • 高中教辅汇总【35GB】
  • Java:Object操作
  • 【课程总结】day29:大模型之深入了解Retrievers解析器
  • 【C/C++】 秋招常考面试题最全总结(让你有一种相见恨晚的感觉)
  • spring如何解决循环依赖
  • 嵌入式linux裸机调试之windows、linux联合gdb
  • Gromacs——使用过程中暴露问题分析及学习
  • docker-文件复制(docker cp:用于在Docker主机和容器之间拷贝文件或目录)
  • 数学建模研赛总结
  • 【Linux】tar 压缩使用绝对路径时解压会出现多级文件夹
  • 新手教学系列——用 VSCode 实现高效远程开发
  • linux查看进程所在的目录
  • 77. Combinations
  • CentOS 7 修改主机名
  • ES6, React, Redux, Webpack写的一个爬 GitHub 的网页
  • exif信息对照
  • iOS 颜色设置看我就够了
  • Java 实战开发之spring、logback配置及chrome开发神器(六)
  • JavaScript 基本功--面试宝典
  • Less 日常用法
  • Python十分钟制作属于你自己的个性logo
  • Selenium实战教程系列(二)---元素定位
  • 大快搜索数据爬虫技术实例安装教学篇
  • 关于extract.autodesk.io的一些说明
  • 极限编程 (Extreme Programming) - 发布计划 (Release Planning)
  • 全栈开发——Linux
  • 一加3T解锁OEM、刷入TWRP、第三方ROM以及ROOT
  • 函数计算新功能-----支持C#函数
  • 曜石科技宣布获得千万级天使轮投资,全方面布局电竞产业链 ...
  • ​探讨元宇宙和VR虚拟现实之间的区别​
  • # dbt source dbt source freshness命令详解
  • # include “ “ 和 # include < >两者的区别
  • (C)一些题4
  • (day 2)JavaScript学习笔记(基础之变量、常量和注释)
  • (done) Go 语言:三种多文件协作方式
  • (M)unity2D敌人的创建、人物属性设置,遇敌掉血
  • (二)fiber的基本认识
  • (个人笔记质量不佳)SQL 左连接、右连接、内连接的区别
  • (十六)视图变换 正交投影 透视投影
  • (源码版)2024美国大学生数学建模E题财产保险的可持续模型详解思路+具体代码季节性时序预测SARIMA天气预测建模
  • .net core 3.0 linux,.NET Core 3.0 的新增功能
  • .net core开源商城系统源码,支持可视化布局小程序
  • .Net 代码性能 - (1)
  • .NET 直连SAP HANA数据库
  • .net/c# memcached 获取所有缓存键(keys)
  • .NET/C# 使用反射注册事件
  • .net程序集学习心得
  • ?php echo ?,?php echo Hello world!;?
  • @TableId注解详细介绍 mybaits 实体类主键注解
  • [ 云计算 | Azure 实践 ] 在 Azure 门户中创建 VM 虚拟机并进行验证
  • [2019/05/17]解决springboot测试List接口时JSON传参异常
  • [AndroidStudio]_[初级]_[修改虚拟设备镜像文件的存放位置]