【iOS】分类、扩展、关联对象

分类、扩展、关联对象

  • 前言
  • 分类
  • 扩展
  • 扩展和分类的区别
  • 关联对象
    • key的几种用法
    • 流程
  • 总结

前言

最近的学习中笔者发现自己对于分类、扩展相关知识并不是很熟悉,刚好看源码类的加载过程中发现有类扩展与关联对象详解。本篇我们来探索一下这部分相关知识,首先我们要记住扩展是编译时就被添加在类中,而分类是在运行时才被整合到类信息中来的

分类

这里我们先来看看使用Clang编译之后,分类的底层结构struct category_t

在这里插入图片描述

这里我们来看看其中的内容,根据名称我们可以发现其中存储了类指针、实例方法表、类方法表、协议表、属性列表,但是并没有类中有的成员变量表。其实看到这里我们就可以明白,我们不可以在分类中定义成员变量,原因很简单,这里面都没有成员变量表

这里还有一个结论:分类可以声明属性,并可以生成对应的set、get方法,但没有去实现该方法

分类加载流程:

  • 在编译阶段将分类中的方法、属性等编译到一个数据结构category_t
  • 将分类中的方法、属性等合并到一个大数组中去,而后参加编译的分类就会在数组前面
  • 将合并后的分类数据插入到原有数据的前面

故而当分类中的方法与原始类中方法重名的时候,会先去调用分类中实现的方法。

扩展

@interface Person ()@property (nonatomic, assign) NSInteger age;  // 私有属性- (BOOL)validateAge;  // 私有方法声明@end

这里我们将一个扩展直接使用Clang转化位cpp文件,我们可以看到其直接被存储到了成员变量表中,同时方法也直接被添加到了metholist中:

在这里插入图片描述

在这里插入图片描述

故而扩展是在编译阶段与该类同时编译的,是类的一部分。扩展中声明的方法只能在该类的@implementation中实现。所以这也就意味着我们无法对系统的类使用扩展。

扩展和分类的区别

类别、分类

  • 专门用来给类添加新的方法
  • 不能给类添加成员属性,添加了成员属性也无法取到
    • 注意:其实可以通过runtime 给分类添加属性,即属性关联,重写settergetter方法
  • 分类中用@property 定义变量,只会生成变量的settergetter方法的声明不能生成方法实现和带下划线的成员变量

扩展

  • 可以说成是特殊的分类,也可称作匿名分类
  • 可以给类添加成员属性,但是是私有变量
  • 可以给类添加方法,也是私有方法

关联对象

这里我们来讲解一下如何通过runtime来给分类添加属性,这里主要分为两部分:

  • 通过objc_setAssociatedObject设值流程
  • 通过objc_getAssociatedObject取值流程

在这里插入图片描述

流程如上所示

我们先来看看取值流程:

objc_setAssociatedObject(id _Nonnull object, const void * _Nonnull key,id _Nullable value, objc_AssociationPolicy policy)
  • 参数一:要关联的对象,即为谁添加关联属性
  • 参数二:标识符,方便下次查找
  • 参数三:value
  • 参数四:属性的策略,即nonatomic、atomic、assign等,下面展示一下所有关联对象的属性类型:

在这里插入图片描述

下面我们来看看objc_setAssociatedObject的源码实现:

在这里插入图片描述

下面我们进入_object_set_associative_reference源码实现来看看:

void
_object_set_associative_reference(id object, const void *key, id value, uintptr_t policy)
{// This code used to work when nil was passed for object and key. Some code// probably relies on that to not crash. Check and handle it explicitly.// rdar://problem/44094390if (!object && !value) return;if (object->getIsa()->forbidsAssociatedObjects())_objc_fatal("objc_setAssociatedObject called on instance (%p) of class %s which does not allow associated objects", object, object_getClassName(object));//object封装成一个数组结构类型,类型为DisguisedPtrDisguisedPtr<objc_object> disguised{(objc_object *)object};//相当于包装了一下 对象object,便于使用// 包装一下 policy - valueObjcAssociation association{policy, value};// retain the new value (if any) outside the lock.association.acquireValue();//根据策略类型进行处理bool isFirstAssociation = false;{//初始化manager变量,相当于自动调用AssociationsManager的析构函数进行初始化AssociationsManager manager;//并不是全场唯一,构造函数中加锁只是为了避免重复创建,在这里是可以初始化多个AssociationsManager变量的AssociationsHashMap &associations(manager.get());//AssociationsHashMap 全场唯一if (value) {auto refs_result = associations.try_emplace(disguised, ObjectAssociationMap{});//返回结构为一个类对if (refs_result.second) {//判断第二个存不存在,即bool值是否为true/* it's the first association we make */isFirstAssociation = true;}/* establish or replace the association */auto &refs = refs_result.first->second;//得到一个空的桶子,找到引用对象类型,即第一个元素的second值auto result = refs.try_emplace(key, std::move(association));//查找当前的key是否有association关联对象if (!result.second) {//如果结果不存在association.swap(result.first->second);}} else {//如果传的是空值,则移除关联,相当于移除auto refs_it = associations.find(disguised);if (refs_it != associations.end()) {auto &refs = refs_it->second;auto it = refs.find(key);if (it != refs.end()) {association.swap(it->second);refs.erase(it);if (refs.size() == 0) {associations.erase(refs_it);}}}}}// Call setHasAssociatedObjects outside the lock, since this// will call the object's _noteAssociatedObjects method if it// has one, and this may trigger +initialize which might do// arbitrary stuff, including setting more associated objects.if (isFirstAssociation)object->setHasAssociatedObjects();// release the old value (outside of the lock).association.releaseHeldValue();//释放
}

我们来看看这段源码的实现过程:

  • 首先检查对象所属类是否禁止关联对象(系统类就不可以),若禁止则直接触发崩溃
  • 创建一个全局管理关联对象的AssociationsManager管理类,并获取唯一的全局静态哈希Map:AssociationsHashMap
  • value是否存在:
  • 若存在,通过try_emplace方法,创建一个空的ObjectAssociationMap去取查询键值对
  • 如果发现没有这个 key 就插入一个空的 BucketT进去并返回true
  • 通过setHasAssociatedObjects方法标记对象存在关联对象即置isa指针的has_assoc属性为true
  • 用当前policy 和 value组成了一个ObjcAssociation替换原来BucketT 中的空
  • 标记一下 ObjectAssociationMap 的第一次为 false

AssociationsManager

我们先来看看其源码实现

在这里插入图片描述

这里我们可以看到AssociationsHashMap从静态变量中取出,所以全场唯一

下面我们来看看这AssociationsHashMap以及ObjectAssociationMap的定义

在这里插入图片描述

这里先说一下DenseMap,这个东西时LLVM实现的高性能哈希表,支持快速插入、查找、删除(笔者具体也不会)

  • 先来看看ObjectAssociationMap,他对应的是一个对象的关联属性集合,通过健快速定位到具体的ObjcAssociation

在这里插入图片描述

​ 这里展示一下该结构体内部包含的内容:关联值的引用计数策略与实际值

  • 再来看看AssociationsHashMap,这是一个全局管理所有对象的关联属性的集合,这里键为伪装指针(DisguisedPtr,值为该对象关联属性表ObjectAssociationMap

这里附一张图来讲解这几个表之间的关系

在这里插入图片描述

下面来说一下这几个map之间的联系与不同:

  • AssociationsManager可以有很多个,但是AssociationsHashMap类型的map只能有一个,是通过AssociationsManager来获取的
  • 这个map中有很多个ObjectAssociationMap类型的map,在上文中的讲解中,我们可以明白每个对象都有一个ObjcAssociation,所以每个对象都会有一个自己的ObjectAssociationMap类型的map

key的几种用法

  • 使用的get方法的@selector作为key
objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, @selector(getter))
  • 使用指针的地址作为key
static void *MyKey = &MyKey;
objc_setAssociatedObject(obj, MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, MyKey)
  • 使用static作为key
static char MyKey;objc_setAssociatedObject(obj, &MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, &MyKey)
  • 使用属性名作为key
objc_setAssociatedObject(obj, @“property”, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_getAssociatedObject(obj, @“property”);

流程

  • 设置关联对象:

    • 调用objc_setAssociatedObject

    • AssociationsManager查找或创建与目标对象相关的ObjectAssociationMap

    • ObjectAssociationMap中查找或创建对应的 ObjcAssociation

    • 将关联值和存储策略设置到 ObjcAssociation 中。

  • 获取关联对象:

    • 调用objc_setAssociatedObject

    • AssociationsManager查找与目标对象相关的ObjectAssociationMap

    • ObjectAssociationMap中查找对应的 ObjcAssociation

    • 返回ObjcAssociation中存储的关联值

  • 移除关联对象:

    • 调用 objc_removeAssociatedObjectsobjc_setAssociatedObject 设置为 nil。
    • AssociationsManager 查找与目标对象相关的ObjectAssociationMap
    • ObjectAssociationMap 中移除对应的 ObjcAssociation
    • 如果ObjectAssociationMap为空,可能会移除整个映射以释放资源。

这个流程其实也就是上文中_object_set_associative_reference的流程,笔者认为这样理解更好一些,下面再附一张图帮助理解

在这里插入图片描述

总结

关联对象就是一个二层哈希的处理,存取的时候都是两层处理,类似于二维数组:
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

30.第二阶段x64游戏实战-认识网络数据包发送流程

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a;图灵Python学院 上一个内容&#xff1a;29.第二阶段x64游戏实战-技能冷却 发送数据包的方式&#xff08;函数&#xff09;操作系统提供…

【每日一题】【前缀和优化】【前/后缀最值】牛客练习赛139 B/C题 大卫的密码 (Hard Version) C++

牛客练习赛139 B题 大卫的密码 (Easy Version) 牛客练习赛139 C题 大卫的密码 (Hard Version) 大卫的密码 题目背景 牛客练习赛139 题目描述 给定一个 n m n\times m nm的网格图&#xff0c;我们使用 ( i , j ) (i,j) (i,j)表示网格中从上往下数第 i i i行和从左往右数第…

文件夹图像批处理教程

前言 因为经常对图像要做数据清洗&#xff0c;又很费时间去重新写一个&#xff0c;我一直在想能不能写一个通用的脚本或者制作一个可视化的界面对文件夹图像做批量的修改图像大小、重命名、划分数据训练和验证集等等。这里我先介绍一下我因为写过的一些脚本&#xff0c;然后我…

【Unity实战笔记】第二十四 · 使用 SMB+Animator 实现基础战斗系统

转载请注明出处&#xff1a;&#x1f517;https://blog.csdn.net/weixin_44013533/article/details/146409453 作者&#xff1a;CSDN|Ringleader| 1 结构 1.1 状态机 1.2 SMB 2 代码实现 2.1 核心控制 Player_Base_SMB 继承 StateMachineBehaviour &#xff0c;控制变量初始…

Python虚拟环境再PyCharm中自由切换使用方法

Python开发中的环境隔离是必不可少的步骤,通过使用虚拟环境可以有效地管理不同项目间的依赖,避免包冲突和环境污染。虚拟环境是Python官方提供的一种独立运行环境,每个项目可以拥有自己单独的环境,不同项目之间的环境互不影响。在日常开发中,结合PyCharm这样强大的IDE进行…

大模型智能体入门扫盲——基于camel的概述

前言 本篇博客想带读者进行一个智能体入门扫盲&#xff0c;了解基础知识&#xff0c;为什么用camel呢&#xff0c;因为小洛发现它们文档对这种智能体的基本组件介绍得很全面深入。 基础概念 agent 一个典型的agent智能体包含三个核心部分&#xff1a; 感知模块&#xff1…

目标检测 RT-DETR(2023)详细解读

文章目录 主干网络&#xff1a;Encoder&#xff1a;不确定性最小Query选择Decoder网络&#xff1a; 将DETR扩展到实时场景&#xff0c;提高了模型的检测速度。网络架构分为三部分组成&#xff1a;主干网络、混合编码器、带有辅助预测头的变换器编码器。具体来说&#xff0c;先利…

DeepSeek 赋能数字农业:从智慧种植到产业升级的全链条革新

目录 一、数字农业的现状与挑战二、DeepSeek 技术解析2.1 DeepSeek 的技术原理与优势2.2 DeepSeek 在人工智能领域的地位与影响力 三、DeepSeek 在数字农业中的应用场景3.1 精准种植决策3.2 病虫害监测与防治3.3 智能灌溉与施肥管理3.4 农产品质量追溯与品牌建设 四、DeepSeek …

<uniapp><vuex><状态管理>在uniapp中,如何使用vuex实现数据共享与传递?

前言 本专栏是基于uniapp实现手机端各种小功能的程序&#xff0c;并且基于各种通讯协议如http、websocekt等&#xff0c;实现手机端作为客户端&#xff08;或者是手持机、PDA等&#xff09;&#xff0c;与服务端进行数据通讯的实例开发。 发文平台 CSDN 环境配置 系统&…

高速串行差分信号仿真分析及技术发展挑战续

7.3 3.125Gbps 差分串行信号设计实例仿真分析 7.3.1 设计用例说明 介绍完 Cadence 系统本身所具有的高速差分信号的仿真分析功能之后&#xff0c;我们以一个实例来说明 3.125Gbps 以下的高速差分系统的仿真分析方法。 在网上下载的设计文件“Booksi_Demo_Allegro160_Finishe…

【Golang】部分语法格式和规则

1、时间字符串和时间戳的相互转换 func main() {t1 : int64(1546926630) // 外部传入的时间戳&#xff08;秒为单位&#xff09;&#xff0c;必须为int64类型t2 : "2019-01-08 13:50:30" // 外部传入的时间字符串//时间转换的模板&#xff0c;golang里面只能是 &quo…

第十六章:数据治理之数据架构:数据模型和数据流转关系

本章我们说一下数据架构&#xff0c;说到数据架构&#xff0c;就很自然的想到企业架构、业务架构、软件架构&#xff0c;因为个人并没有对这些内容进行深入了解&#xff0c;所以这里不做比对是否有相似或者共通的地方&#xff0c;仅仅来说一下我理解的数据架构。 1、什么是架构…

Day126 | 灵神 | 二叉树 | 层数最深的叶子结点的和

Day126 | 灵神 | 二叉树 | 层数最深的叶子结点的和 1302.层数最深的叶子结点的和 1302. 层数最深叶子节点的和 - 力扣&#xff08;LeetCode&#xff09; 思路&#xff1a; 这道题用层序遍历的思路比较好想&#xff0c;就把每层的都算一下&#xff0c;然后返回最后一层的和就…

PCIE 4.0 vs PCIE 5.0固态硬盘——区别、科普与选购场景全解析

随着数字内容和高性能计算需求的爆发&#xff0c;固态硬盘&#xff08;SSD&#xff09;已成为PC、游戏主机和工作站不可或缺的核心硬件。面对市面上层出不穷的新一代SSD产品&#xff0c;大家最常见的一个疑惑&#xff1a;**PCIe 4.0和PCIe 5.0固态硬盘&#xff0c;到底有啥区别…

vue pinia 独立维护,仓库统一导出

它允许您跨组件/页面共享状态 持久化 安装依赖pnpm i pinia-plugin-persistedstate 将插件添加到 pinia 实例上 pinia独立维护 统一导出 import { createPinia } from pinia import piniaPluginPersistedstate from pinia-plugin-persistedstateconst pinia creat…

Dify源码学习

文章目录 1 大模型基本原理1.1 model_context_tokens、max_tokens和prompt_tokens1.1.1 三者之间的关系1.1.2 总结对比 2 Dify源代码2.0 前后端代码跑起来【0】准备开发环境【1】下载代码【2】运行后端&#xff08;1&#xff09;Start the docker-compose stack&#xff08;2&a…

连接表、视图和存储过程

1. 视图 1.1. 视图的概念 视图&#xff08;View&#xff09;&#xff1a;虚拟表&#xff0c;本身不存储数据&#xff0c;而是封装了一个 SQL 查询的结果集。 用途&#xff1a; 只显示部分数据&#xff0c;提高数据访问的安全性。简化复杂查询&#xff0c;提高复用性和可维护…

微信小程序中,解决lottie动画在真机不显示的问题

api部分 export function getRainInfo() {return onlineRequest({url: /ball/recruit/getRainInfo,method: get}); }data存储json数据 data&#xff1a;{rainJson:{} }onLoad方法获取json数据 onLoad(options) {let that thisgetRainInfo().then((res)>{that.setData({r…

从加密到信任|密码重塑车路云一体化安全生态

目录 一、密码技术的核心支撑 二、典型应用案例 三、未来发展方向 总结 车路云系统涉及海量实时数据交互&#xff0c;包括车辆位置、传感器信息、用户身份等敏感数据。其安全风险呈现三大特征&#xff1a; 开放环境威胁&#xff1a;V2X&#xff08;车与万物互联&#xff0…

光谱相机在地质勘测中的应用

一、‌矿物识别与蚀变带分析‌ ‌光谱特征捕捉‌ 通过可见光至近红外&#xff08;400-1000nm&#xff09;的高光谱分辨率&#xff08;可达3.5nm&#xff09;&#xff0c;精确识别矿物的“光谱指纹”。例如&#xff1a; ‌铜矿‌&#xff1a;在400-500nm波段反射率显著低于围…