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

server/2024/9/24 5:01:36/

文章目录

    • 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

http://www.ppmy.cn/server/121176.html

相关文章

小柴冲刺软考中级嵌入式系统设计师系列一、计算机系统基础知识(6)可靠性与系统性能评测基础

目录 1、计算机可靠性 串联系统 并联系统 2、计算机系统的性能评价 性能评测的常用方法 基准测试程序 flechazohttps://www.zhihu.com/people/jiu_sheng 小柴冲刺嵌入式系统设计师系列总目录https://blog.csdn.net/qianshang52013/article/details/139975720?spm1001.2…

.NET 控制台应用程序连接 MySQL 数据库实现增删改查

概述 本文旨在指导开发者如何通过.NET控制台应用程序与MySQL数据库进行交互,从而执行基本的数据操作:创建(增加)、读取(查询)、更新以及删除记录。这样的技能对于任何需要数据持久化的应用程序来说都是基础且重要的,无论是在Web应用还是桌面软件中都非常有用。我们将使…

2024蓝桥杯省B好题分析

题解来自洛谷&#xff0c;作为学习 目录 宝石组合 数字接龙 爬山 拔河 宝石组合 # [蓝桥杯 2024 省 B] 宝石组合## 题目描述在一个神秘的森林里&#xff0c;住着一个小精灵名叫小蓝。有一天&#xff0c;他偶然发现了一个隐藏在树洞里的宝藏&#xff0c;里面装满了闪烁着美…

react + antDesign封装图片预览组件(支持多张图片)

需求场景&#xff1a;最近在开发后台系统时经常遇到图片预览问题&#xff0c;如果一个一个的引用antDesign的图片预览组件就有点繁琐了&#xff0c;于是在antDesign图片预览组件的基础上二次封装了一下&#xff0c;避免重复无用代码的出现 效果 公共预览组件代码 import React…

2024从传统到智能,AI做PPT软件的崛起之路

随着AI技术的飞速进步&#xff0c;它已悄然渗透至我们的工作与学习之中&#xff0c;不仅助力写作与绘画创作&#xff0c;就连PPT制作这一传统办公领域也迎来了AI的革新。我最近有幸探索了一系列AI驱动的PPT制作工具&#xff0c;亲身体验后深感震撼——合理利用这些ai做ppt工具&…

Tcping:一款实用的端口存活检测工具

简介 tcping 是一个基于TCP协议的网络诊断工具,通过发送 TCP SYN/ACK包来检测目标主机的端口状态。 官网:tcping.exe - ping over a tcp connection 优点: (1)监听服务器端口状态:tcping 可以检测指定端口的状态,默认是80端口,也可以指定其他端口。 (2)显示ping返…

MySQL深度探索:掌握触发器自动化与精细用户权限管理,提升数据库效能与安全

作者简介&#xff1a;我是团团儿&#xff0c;是一名专注于云计算领域的专业创作者&#xff0c;感谢大家的关注 座右铭&#xff1a; 云端筑梦&#xff0c;数据为翼&#xff0c;探索无限可能&#xff0c;引领云计算新纪元 个人主页&#xff1a;团儿.-CSDN博客 目录 前言&#x…

uniapp打字效果流式输出

在UniApp中实现打字效果的流式输出&#xff0c;可以按照以下思路进行&#xff1a; 1. 定义数据结构&#xff1a; 创建一个data对象&#xff0c;包含完整文本、当前显示文本和字符索引。 2. 使用生命周期钩子&#xff1a; 在mounted钩子中启动打字效果的逻辑。 3. 编写打字…