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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/883668.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SentencePiece进行文本分类

SentencePieces 前言 Step1:故事 SentencePiece 是一个无监督的文本分词器和 detokenizer(还原回去的&#xff1f;)主要用于词汇表大小是预定的文本生成系统中它拓展了原始句子的训练&#xff0c;实现子词单元如 BPE 和 unigram language model技术亮点 纯数据驱动&#xff…

那年我双手插兜,使用IPv6+DDNS动态域名解析访问NAS

估计有很多科技宅和我一样&#xff0c;会买一个NAS存储或者自己折腾刷一下黑群晖玩玩&#xff0c;由于运营商不给分配固定的公网IP&#xff0c;就导致我在外出的时候无法访问家里的NAS&#xff0c;于是远程访问常常受到IP地址频繁变动的困扰。为了解决这一问题&#xff0c;结合…

【HTTP】请求“报头”,Referer 和 Cookie

Referer 描述了当前这个页面是从哪里来的&#xff08;从哪个页面跳转过来的&#xff09; 浏览器中&#xff0c;直接输入 URL/点击收藏夹打开的网页&#xff0c;此时是没有 referer。当你在 sogou 页面进行搜索时&#xff0c;新进入的网页就会有 referer 有一个非常典型的用…

gitlab默认克隆地址的修改

目录 1.找到opt/gitlab/embedded/service/gitlab-rails/config目录&#xff0c;打开gitlab.yml 2.修改地址和端口 3.重启gitlab 1.找到opt/gitlab/embedded/service/gitlab-rails/config目录&#xff0c;打开gitlab.yml cd /opt/gitlab/embedded/service/gitlab-rails/confi…

jmeter断言---响应断言

请求http://www.baidu.com 检查&#xff1a;让程序检查响应数据中是否包含“百度一下&#xff0c;你就知道” 操作步骤&#xff1a; 1.添加线程组 2.添加http请求 3.添加断言&#xff08;需要在http请求下添加断言&#xff0c;而且可以根据断言测试字段等信息新建不同的断…

docker-图形化工具-portainer的使用

文章目录 1、安装和启动2、设置登陆密码3、dashboard 上述对容器和镜像的管理都是基于docker客户端的命令来完成&#xff0c;不太方便。为了方便的对docker中的一些对象(镜像、容器、数据卷…)来进行管理&#xff0c;可以使用Portainer来完成。Portainer是一个可视化的容器镜像…

【RabbitMQ】RabbitMQ 的概念以及使用RabbitMQ编写生产者消费者代码

目录 1. RabbitMQ 核心概念 1.1生产者和消费者 1.2 Connection和Channel 1.3 Virtual host 1.4 Queue 1.5 Exchange 1.6 RabbitMO工作流程 2. AMQP 3.RabbitMO快速入门 3.1.引入依赖 3.2.编写生产者代码 ​3.3.编写消费者代码 4.源码 1. RabbitMQ 核心概念 在安装…

LiveNVR监控流媒体Onvif/RTSP功能-支持电子放大拉框放大直播视频拉框放大录像视频流拉框放大电子放大

LiveNVR监控流媒体Onvif/RTSP功能-支持电子放大拉框放大直播视频拉框放大录像视频流拉框放大电子放大 1、视频广场2、录像回看3、RTSP/HLS/FLV/RTMP拉流Onvif流媒体服务 1、视频广场 视频广场 -》播放 &#xff0c;左键单击可以拉取矩形框&#xff0c;放大选中的范围&#xff…

NLP-transformer学习:(7)evaluate实践

NLP-transformer学习&#xff1a;&#xff08;7&#xff09;evaluate 使用方法 打好基础&#xff0c;为了后面学习走得更远。 本章节是单独的 NLP-transformer学习 章节&#xff0c;主要实践了evaluate。同时&#xff0c;最近将学习代码传到&#xff1a;https://github.com/Mex…

STL之vector篇(下)(手撕底层代码,从零实现vector的常用指令,深度剖析并优化其核心代码)

文章目录 1.基本结构与初始化1.1 空构造函数的实现与测试1.2 带大小和默认值的构造函数1.3 使用迭代器范围初始化的构造函数(建议先看完后面的reserve和push_back)1.4 拷贝构造函数1.5 赋值操作符的实现&#xff08;深拷贝&#xff09;1.6 析构函数1.7 begin 与 end 迭代器 2. …

使用宝塔部署项目在win上

项目部署 注意&#xff1a; 前后端部署项目&#xff0c;需要两个域名&#xff08;二级域名&#xff0c;就是主域名结尾的域名&#xff0c;需要在主域名下添加就可以了&#xff09;&#xff0c;前端一个&#xff0c;后端一个 思路&#xff1a;访问域名就会浏览器会加载前端的代…

如何守护变美神器安全?红外热像仪:放开那根美发棒让我来!

随着智能家电市场的迅速发展&#xff0c;制造商们越来越关注生产过程中效率和质量的提升。如何守护变美神器安全&#xff1f;红外热像仪&#xff1a;放开那根卷发棒让我来&#xff01; 美发棒生产遇到什么困境&#xff1f; 美发棒生产过程中会出现设备加热不均情况&#xff0c…

[数据库实验五] 审计及触发器

一、实验目的与要求&#xff1a; 1.了解MySQL审计功能及实现方式 2.掌握触发器的工作原理、定义及操作方法 二、实验内容&#xff1a; 注&#xff1a; 在同一个触发器内编写多行代码&#xff0c;需要用结构begin ……end 函数current_user()获得当前登录用户名 1.自动保存…

智慧城市主要运营模式分析

(一)运营模式演变 作为新一代信息化技术落地应用的新事物,智慧城市在建设模式方面借鉴了大量工程建设的经验,如平行发包(DBB,Design-Bid-Build)、EPC工程总承包、PPP等模式等,这些模式在不同的发展阶段和条件下发挥了重要作用。 在智慧城市发展模式从政府主导、以建为主、…

linux----进程地址空间

前言 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、空间分布 二、栈和堆的特点 &#xff08;1&#xff09;栈堆相对而生&#xff0c;堆是向上增长的&#xff0c;栈是向下增长的。 验证&#xff1a;堆是向上增长的 这里我们看到申请的堆&#xff…

记一次Windows状态栏不显示问题

文章目录 &#x1fa9f;解决方案☁️单次处理☁️有效处理 &#x1fa9f;现象&#x1fa9f;尝试的操作⭐END&#x1f31f;跋&#x1f31f;交流方式 &#x1fa9f;解决方案 ☁️单次处理 重启explorer.exe 命令行操作 注意&#xff0c;使用命令行操作的时候&#xff0c;出现…

链动 2+1 模式 S2B2C 商城小程序源码:创新价格盈利模式探索

摘要&#xff1a;本文深入探讨了价格盈利模式的两种类型&#xff0c;即价格返利模式和动态定价盈利模式。通过引入链动 21 模式 S2B2C 商城小程序源码&#xff0c;分析其在实现这两种价格盈利模式方面的优势和应用场景&#xff0c;为朋友圈卖货及电商领域的发展提供新的思路和方…

QT菜单之快捷菜单设计

快捷菜单又称为上下文菜单&#xff0c;通常在用鼠标右击的时候弹出。创建快捷菜单的方法和创建菜单栏菜单类似。 效果图&#xff1a; 一、将MainWindow类对象的ContextMenuPolicy属性设置为customContextMenu。 打开mainWindow.ui&#xff0c;在属性视图上找到ContextMenuPoli…

一文掌握python单元测试unittest(二)

接上篇:https://blog.csdn.net/qq_38120851/article/details/141642215 目录 四、参数化测试 1、使用 subTest 2、使用装饰器 3)使用第三方库parameterized 五、跳过测试 1、使用 unittest.skip() 或 unittest.skipIf() 装饰器: 2、使用 setUp() 方法中的断言来跳过整…

EasyCVR智慧公园视频智能管理方案:赋能公园安全管理新高度

随着城市化进程的加速&#xff0c;智慧城市建设已成为提升城市管理效率、增强居民生活质量的重要途径。智慧公园作为智慧城市的重要组成部分&#xff0c;其安全与管理水平直接影响着市民的休闲娱乐体验。EasyCVR智慧公园视频智能管理方案&#xff0c;正是基于这一背景应运而生&…