【iOS】——属性关键字的底层原理
strong,retain,copy,atomic,nonatomic c++源码
@interface propertyTest : NSObject
@property (nonatomic, strong) NSString *nsstring___StrongTest;
@property (nonatomic, retain) NSString *nsstring___RetainTest;
@property (nonatomic, copy) NSString *nsstring___CopyTest;
@property (atomic, copy) NSString *nsstring___AtomicCopyTest;
@end
用clang -rewrite-objc propertyTest.m生成c++源码看如下:
// 实现propertyTest类的定义
@implementation propertyTest// 以下是对不同属性类型的实现细节// 对于强引用属性(__strong)的get方法
static NSString * _I_propertyTest_nsstring___StrongTest(propertyTest * self, SEL _cmd) {// 强引用的get方法直接返回实例变量的值。// 使用*(NSString **)来解引用指针,((char *)self + ...)计算出实例变量在内存中的位置。return (*(NSString **)((char *)self + OBJC_IVAR_$_propertyTest$_nsstring___StrongTest));
}// 对于强引用属性(__strong)的set方法
static void _I_propertyTest_setNsstring___StrongTest_(propertyTest * self, SEL _cmd, NSString *nsstring___StrongTest) {// 强引用的set方法直接赋值给实例变量。(*(NSString **)((char *)self + OBJC_IVAR_$_propertyTest$_nsstring___StrongTest)) = nsstring___StrongTest;
}// 对于retain属性的get方法
static NSString * _I_propertyTest_nsstring___RetainTest(propertyTest * self, SEL _cmd) {// retain属性的get方法同样直接返回实例变量的值。return (*(NSString **)((char *)self + OBJC_IVAR_$_propertyTest$_nsstring___RetainTest));
}// 声明objc_setProperty函数,这是一个外部定义的运行时函数。
extern "C" __declspec(dllimport) void objc_setProperty (id, SEL, long, id, bool, bool);// 对于retain属性的set方法
static void _I_propertyTest_setNsstring___RetainTest_(propertyTest * self, SEL _cmd, NSString *nsstring___RetainTest) {// retain属性的set方法使用objc_setProperty运行时函数,该函数负责调用retain和release。// 第四个参数为false表示不需要复制,第五个参数为false表示不需要原子性操作。objc_setProperty (self, _cmd, __OFFSETOFIVAR__(struct propertyTest, _nsstring___RetainTest), (id)nsstring___RetainTest, false, false);
}// 对于copy属性的get方法
static NSString * _I_propertyTest_nsstring___CopyTest(propertyTest * self, SEL _cmd) {// copy属性的get方法直接返回实例变量的值。return (*(NSString **)((char *)self + OBJC_IVAR_$_propertyTest$_nsstring___CopyTest));
}// 对于copy属性的set方法
static void _I_propertyTest_setNsstring___CopyTest_(propertyTest * self, SEL _cmd, NSString *nsstring___CopyTest) {// copy属性的set方法使用objc_setProperty运行时函数,该函数负责调用copy和release。// 第四个参数为false表示不需要retain,第五个参数为true表示需要复制。objc_setProperty (self, _cmd, __OFFSETOFIVAR__(struct propertyTest, _nsstring___CopyTest), (id)nsstring___CopyTest, false, true);
}// 声明objc_getProperty函数,这是一个外部定义的运行时函数。
extern "C" __declspec(dllimport) id objc_getProperty(id, SEL, long, bool);// 对于原子性copy属性(atomic copy)的get方法
static NSString * _I_propertyTest_nsstring___AtomicCopyTest(propertyTest * self, SEL _cmd) {// 原子性copy属性的get方法使用objc_getProperty运行时函数,该函数负责原子性读取。// 第四个参数为true表示需要原子性操作。typedef NSString * _TYPE;return (_TYPE)objc_getProperty(self, _cmd, __OFFSETOFIVAR__(struct propertyTest, _nsstring___AtomicCopyTest), true);
}// 对于原子性copy属性(atomic copy)的set方法
static void _I_propertyTest_setNsstring___AtomicCopyTest_(propertyTest * self, SEL _cmd, NSString *nsstring___AtomicCopyTest) {// 原子性copy属性的set方法使用objc_setProperty运行时函数,该函数负责调用copy和release,同时保证原子性。// 第四个参数为true表示需要原子性操作,第五个参数为true表示需要复制。objc_setProperty (self, _cmd, __OFFSETOFIVAR__(struct propertyTest, _nsstring___AtomicCopyTest), (id)nsstring___AtomicCopyTest, true, true);
}// 结束propertyTest类的实现
@end
- strong、copy、retain 的
get方法是根据地址偏移找到对应的实例变量直接返回 - atomic的
get方法用到了objc_getProperty方法 - strong 的
set根据地址偏移找到对应的实例变量直接赋值 - retain、copy、atomic 的
set方法 用到了objc_setProperty
objc_setProperty
void objc_setProperty(id self, SEL _cmd, ptrdiff_t offset, id newValue, BOOL atomic, signed char shouldCopy)
{objc_setProperty_non_gc(self, _cmd, offset, newValue, atomic, shouldCopy);
}
这个方法有六个参数,self指对象自身;_cmd是方法选择器;ptrdiff_t 是一个标准整型,用于表示指针之间的差值。offset是实例变量的偏移量;newValue用来赋值给属性的新值;atomic表示是否需要原子性操作;shouldCopy 代表是否应该复制属性值,如果为1,则执行拷贝操作。
objc_setProperty调用了objc_setProperty_non_gc
void objc_setProperty_non_gc(id self, SEL _cmd, ptrdiff_t offset, id newValue, BOOL atomic, signed char shouldCopy)
{//判断是否需要进行普通拷贝。如果 shouldCopy 非零且不等于 MUTABLE_COPY,那么需要拷贝 newValue。bool copy = (shouldCopy && shouldCopy != MUTABLE_COPY);// 判断是否需要进行可变拷贝(深拷贝)。如果 shouldCopy 等于 MUTABLE_COPY,那么需要可变拷贝 newValue。bool mutableCopy = (shouldCopy == MUTABLE_COPY);reallySetProperty(self, _cmd, newValue, offset, atomic, copy, mutableCopy);
}
这里调用了 reallySetProperty, 此函数会根据提供的参数执行属性值的设置,包括处理原子性操作和拷贝行为。该函数的最后两个参数由shouldCopy决定。
-
如果shouldCopy=0 那么 copy = NO,mutableCopy = NO
-
如果shouldCopy=1 那么 copy = YES,mutableCopy = NO
-
如果shouldCopy=MUTABLE_COPY 那么 copy = NO,mutableCopy = YES
reallySetProperty
static inline void reallySetProperty(id self, SEL _cmd, id newValue, ptrdiff_t offset, bool atomic, bool copy, bool mutableCopy)
{// 检查 offset 是否为零。如果是零,则这意味着我们正在设置类而不是属性。if (offset == 0) {// 使用 object_setClass 函数来改变对象的类。object_setClass(self, newValue);return;}// 用于存储旧属性值的变量。id oldValue;// 计算属性槽的地址,这是对象中存储属性值的位置。id *slot = (id*) ((char*)self + offset);// 如果 copy 为真,则调用 copyWithZone: 方法创建一个新对象作为属性值。if (copy) {newValue = [newValue copyWithZone:nil];} // 如果 mutableCopy 为真,则调用 mutableCopyWithZone: 方法创建一个可变拷贝作为属性值。else if (mutableCopy) {newValue = [newValue mutableCopyWithZone:nil];} // 否则,保留 newValue 的引用计数,除非它已经与现有的属性值相同。else {if (*slot == newValue) return;newValue = objc_retain(newValue);}// 如果设置是非原子的,直接替换旧值。if (!atomic) {oldValue = *slot;*slot = newValue;} // 如果设置是原子的,使用 spinlock 锁来保证线程安全。else {// 获取指向 slot 对应的 spinlock 的引用。spinlock_t& slotlock = PropertyLocks[slot];// 锁定 spinlock。slotlock.lock();oldValue = *slot;*slot = newValue;// 解锁 spinlock。slotlock.unlock();}// 释放旧属性值的引用计数。objc_release(oldValue);
}
- 如果
offset是 0,说明我们不是在设置普通的属性,而是在尝试更改对象的类,这时调用object_setClass函数。 - 计算属性值在对象内存布局中的具体位置
- 如果
copy为真,则调用copyWithZone:方法进行不可变拷贝。 - 如果
mutableCopy为真,则调用mutableCopyWithZone:方法创建一个可变拷贝。 - 如果两者都为假,检查新值是否与旧值相同;如果不相同,保留新值的引用计数。
- 如果
atomic为假,直接用新值替换旧值。 - 如果
atomic为真,使用 spinlock 锁来确保在多线程环境中设置属性的操作是线程安全的。 - 最后,释放旧属性值的引用计数,以便正确管理内存。
从上面的源码可以得出:
copy关键字的实现最终还是调用了copyWithZone方法,用copy修饰的属性,赋值的时候,不管本身是否是可变对象,赋值给属性之后都是不可变对象。
如果copy和mutableCopy都为NO,会调用 objc_retain
retain调用set方法
retain调用set方法时调用的还是reallySetProperty函数,传入的参数为atomic参数为NO,copy参数为NO,mutableCopy也为NO, 最终的实现相当于是:
static inline void reallySetProperty(id self, SEL _cmd, id newValue, ptrdiff_t offset, bool atomic, bool copy, bool mutableCopy)
{id oldValue;id *slot = (id*) ((char*)self + offset);if (*slot == newValue) return;newValue = objc_retain(newValue);oldValue = *slot;*slot = newValue;objc_release(oldValue);
}
copy调用set方法
copy调用的还是reallySetProperty函数,传入的参数为atomic参数为NO,copy参数为YES,mutableCopy为NO, 最终的实现相当于是:
static inline void reallySetProperty(id self, SEL _cmd, id newValue, ptrdiff_t offset, bool atomic, bool copy, bool mutableCopy)
{id oldValue;//取出变量id *slot = (id*) ((char*)self + offset);newValue = [newValue copyWithZone:nil];oldValue = *slot;*slot = newValue;objc_release(oldValue);
}
atomic和copy调用set方法
调用的还是reallySetProperty函数,传入的参数为atomic参数为YES,copy参数为YES,mutableCopy为NO, 最终的实现相当于是:
static inline void reallySetProperty(id self, SEL _cmd, id newValue, ptrdiff_t offset, bool atomic, bool copy, bool mutableCopy)
{id oldValue;id *slot = (id*) ((char*)self + offset);newValue = [newValue copyWithZone:nil];spinlock_t& slotlock = PropertyLocks[slot];slotlock.lock();oldValue = *slot;*slot = newValue; slotlock.unlock();objc_release(oldValue);
}
objc_getProperty
id objc_getProperty(id self, SEL _cmd, ptrdiff_t offset, BOOL atomic)
{return objc_getProperty_non_gc(self, _cmd, offset, atomic);
}
这里跟objc_setProperty方法一样都相当于接口函数,调用更深层的objc_getProperty_non_gc方法
id objc_getProperty_non_gc(id self, SEL _cmd, ptrdiff_t offset, BOOL atomic)
{// 如果 offset 是 0,这通常意味着我们试图获取的是类本身,而非普通属性。if (offset == 0) {// 调用 object_getClass 函数来获取对象的类。return object_getClass(self);}// 计算属性值所在的内存地址。id *slot = (id*) ((char*)self + offset);// 如果设置为非原子操作,直接返回属性槽中的值,不涉及引用计数操作。if (!atomic) return *slot;// 原子操作,需要锁定以防止并发修改。// 获取指向属性槽对应的 spinlock 锁。spinlock_t& slotlock = PropertyLocks[slot];// 锁定 spinlock。slotlock.lock();// 读取属性槽中的值并增加其引用计数,确保即使在读取后被其他线程释放也能安全访问。id value = objc_retain(*slot);// 解锁 spinlock。slotlock.unlock();// 为了性能考虑,我们安全地在 spinlock 外部执行 autorelease,这样可以避免锁的竞争。// objc_autoreleaseReturnValue 将值放入 autorelease pool 中,当 pool 清理时会自动释放。return objc_autoreleaseReturnValue(value);
}
- 如果
offset是 0,这表明我们可能试图获取对象的类,而不是属性值。此时调用object_getClass函数来获取对象的类。 - 计算属性值在对象内存布局中的具体位置
- 如果
atomic为NO,那么函数直接返回属性槽中的值,没有引用计数的增加或减少操作。 - 获取指向属性槽对应的 spinlock 锁,锁定 spinlock
- 读取属性槽中的值,并通过
objc_retain函数增加其引用计数 - 解锁 spinlock,结束对属性值修改的锁定
- 使用
objc_autoreleaseReturnValue函数将value放入 autorelease pool 中,这会在适当的时机自动释放value的引用,从而避免内存泄漏
可以看到atomic为yes的时候,对应属性的set、get方法用到了spinlock_t锁,保证了set、get方法的线程安全,但是并不能保证其他操作的线程安全,比如对属性进行进行release操作。
相关文章:
【iOS】——属性关键字的底层原理
strong,retain,copy,atomic,nonatomic c源码 interface propertyTest : NSObject property (nonatomic, strong) NSString *nsstring___StrongTest; property (nonatomic, retain) NSString *nsstring___RetainTest; property (n…...
电影类平台如何选择服务器
电影类平台如何选择服务器 1、数据存储 电影网站对服务器的要求是比较高的,对存储空间的需求特别大,所以在服务器选择上首先要确保足够大的存储空间。另外,当你的网站内容特别多时,内存不够用,可以选择增加内存&#x…...
递归神经网络(RNN)及其预测和分类的Python和MATLAB实现
递归神经网络(Recurrent Neural Networks,RNN)是一种广泛应用于序列数据建模的深度学习模型。相比于传统的前馈神经网络,RNN具有记忆和上下文依赖性的能力,适用于处理具有时序关联性的数据,如文本、语音、时…...
以flask为后端的博客项目——星云小窝
以flask为后端的博客项目——星云小窝 文章目录 以flask为后端的博客项目——星云小窝前言一、星云小窝项目——项目介绍(一)二、星云小窝项目——项目启动(二)三、星云小窝项目——项目结构(三)四、谈论一…...
CUDA编程02 - 数据并行介绍
一:概述 数据并行是指在数据集的不同部分上执行计算工作,这些计算工作彼此相互独立且可以并行执行。许多应用程序都具有丰富的数据并行性,使其能够改造成可并行执行的程序。因此,对于程序员来说,熟悉数据并行的概念以及使用并行编程语言来编写数据并行的代码是非常重要的。…...
Android 视频音量图标
attrs.xml <?xml version"1.0" encoding"utf-8"?> <resources><!--图标颜色--><attr name"ijkSolid" format"color|reference" /><!--喇叭底座宽度--><attr name"ijkCornerWidth" form…...
VScode 修改 Markdown Preview Enhanced 字体以及大纲编号
修改字体和背景颜色 按快捷键 Ctrl , 打开设置,搜索 markdown-preview-enhanced.previewTheme,选择一个黑色主题的css,如 github-dark.css. 修改自动编号和背景颜色 背景颜色 按 F1 或者 Ctrl Shift P,输入 Customize CSS…...
TCP的FIN报文可否携带数据
问题发现: 发现FTP-DATA数据传输完,TCP的挥手似乎只有两次 实际发现FTP-DATA报文中,TCP层flags中携带了FIN标志 piggyback FIN 问题转化为 TCP packet中如果有FIN flag,该报文还能携带data数据么? 答案是肯定的 RFC7…...
【GoF23种设计模式+简单工厂模式】
一、设计模式概述与类型 1.1、设计模式的一般定义: 设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结,使用设计模式是为了可重用代码,让代码更容易被他人理解并且保证代…...
北醒单点激光雷达更改id和波特率以及Ubuntu20.04下CAN驱动
序言: 需要的硬件以及软件 1、USB-CAN分析仪使用顶配pro版本,带有支持ubuntu下的驱动包的,可以读取数据。 2、电源自备24V电源 3、单点激光雷达接线使用can线可以组网。 一、更改北醒单点激光雷达的id号和波特率 安装并运行USB-CAN分析仪自带…...
【线性代数】矩阵变换
一些特殊的矩阵 一,对角矩阵 1,什么是对角矩阵 表示将矩阵进行伸缩(反射)变换,仅沿坐标轴方向伸缩(反射)变换。 2,对角矩阵可分解为多个F1矩阵,如下: 二&a…...
聚焦智慧出行,TDengine 与路特斯科技再度携手
在全球汽车行业向电动化和智能化转型的过程中,智能驾驶技术正迅速成为行业的焦点。随着消费者对出行效率、安全性和便利性的需求不断提升,汽车制造商们需要在全球范围内实现低延迟、高质量的数据传输和处理,以提升用户体验。在此背景下&#…...
虚拟机迁移报错:虚拟机版本与主机“x.x.x.x”的版本不兼容
1.虚拟机在VCenter上从一个ESXi迁移到另一个ESXi上时报错:虚拟机版本与主机“x.x.x.x”的版本不兼容。 2.例如从10.0.128.13的ESXi上迁移到10.0.128.11的ESXi上。点击10.0.128.10上的任意一台虚拟机,查看虚拟机版本。 3.确认要迁移的虚拟机磁盘所在位…...
【教程】vscode添加powershell7终端
win10自带的 powershell 是1.0版本的,太老了,更换为powershell7后,在 vscode 的集成终端中没有显示本篇教程记录在vscode添加powershell7终端的过程 打开vscode终端配置 然后来到这个页面进行设置 查看 powershell7 的安装位置ÿ…...
如何乘上第四次工业革命的大船
如何乘上第四次工业革命的大船 第四次工业革命通常被认为是信息技术和数字化时代的到来,但具体影响哪些产业,以及它将如何演变和展开,仍然是一个广泛讨论的话题。 然而,已经可以看到一些领域可能受到第四次工业革命的深远影响,例如人工智能、物联网、大数据、生物技术、可…...
RKNN执行bash ./build-linux_RK3566_RK3568.sh 报错
目录 报错信息: 原因分析: 解决办法: 报错信息: CMake Error at /usr/share/cmake-3.22/Modules/CMakeDetermineCCompiler.cmake:49 (message): Could not find compiler set in environment variable CC: aarch64-linux-gnu-gcc. Call Stack (most recent call fir…...
Linux常用命令整理
本文将分享一些常用的Linux命令。根据功能的不同,大概分为以下几个方面,一是文件相关命令,二是进程相关命令,三是网络相关命令,四是磁盘相关命令,五是用户管理相关命令,六是系统命令。 1. 文件…...
python 闭包、装饰器
一、闭包: 1. 外部函数嵌套内部函数 2. 外部函数返回内部函数 3.内部函数可以访问外部函数局部变量 闭包(Closure)是指在一个函数内部定义的函数,并且内部函数可以访问外部函数的局部变量,即使外部函数已经执行…...
[pycharm]解决pycharm运行程序出现卡住scanning files to index索引的问题
有时候会出现索引问题,显示scanning files to index 解决方法: in pycharm, go to the "File" on the left top, then select "invalidate caches/restart...", and press "invalidate and restart". 然后等它自己重启…...
python每日学习11:numpy库的用法(下)
python每日学习11:numpy库的用法(下) 数组的拼接 名方法称说明concatenate连接沿现有轴的数组序列hstack水平堆叠序列中的数组(列方向)vstack竖直堆叠序列中的数组(行方向)concatenate函数用于沿指定轴连接相同形状的两…...
【人工智能】神经网络的优化器optimizer(二):Adagrad自适应学习率优化器
一.自适应梯度算法Adagrad概述 Adagrad(Adaptive Gradient Algorithm)是一种自适应学习率的优化算法,由Duchi等人在2011年提出。其核心思想是针对不同参数自动调整学习率,适合处理稀疏数据和不同参数梯度差异较大的场景。Adagrad通…...
k8s从入门到放弃之Ingress七层负载
k8s从入门到放弃之Ingress七层负载 在Kubernetes(简称K8s)中,Ingress是一个API对象,它允许你定义如何从集群外部访问集群内部的服务。Ingress可以提供负载均衡、SSL终结和基于名称的虚拟主机等功能。通过Ingress,你可…...
【Java学习笔记】Arrays类
Arrays 类 1. 导入包:import java.util.Arrays 2. 常用方法一览表 方法描述Arrays.toString()返回数组的字符串形式Arrays.sort()排序(自然排序和定制排序)Arrays.binarySearch()通过二分搜索法进行查找(前提:数组是…...
【SpringBoot】100、SpringBoot中使用自定义注解+AOP实现参数自动解密
在实际项目中,用户注册、登录、修改密码等操作,都涉及到参数传输安全问题。所以我们需要在前端对账户、密码等敏感信息加密传输,在后端接收到数据后能自动解密。 1、引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId...
《通信之道——从微积分到 5G》读书总结
第1章 绪 论 1.1 这是一本什么样的书 通信技术,说到底就是数学。 那些最基础、最本质的部分。 1.2 什么是通信 通信 发送方 接收方 承载信息的信号 解调出其中承载的信息 信息在发送方那里被加工成信号(调制) 把信息从信号中抽取出来&am…...
人机融合智能 | “人智交互”跨学科新领域
本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...
解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist
现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...
CVPR2025重磅突破:AnomalyAny框架实现单样本生成逼真异常数据,破解视觉检测瓶颈!
本文介绍了一种名为AnomalyAny的创新框架,该方法利用Stable Diffusion的强大生成能力,仅需单个正常样本和文本描述,即可生成逼真且多样化的异常样本,有效解决了视觉异常检测中异常样本稀缺的难题,为工业质检、医疗影像…...
Ubuntu系统复制(U盘-电脑硬盘)
所需环境 电脑自带硬盘:1块 (1T) U盘1:Ubuntu系统引导盘(用于“U盘2”复制到“电脑自带硬盘”) U盘2:Ubuntu系统盘(1T,用于被复制) !!!建议“电脑…...
DeepSeek源码深度解析 × 华为仓颉语言编程精粹——从MoE架构到全场景开发生态
前言 在人工智能技术飞速发展的今天,深度学习与大模型技术已成为推动行业变革的核心驱动力,而高效、灵活的开发工具与编程语言则为技术创新提供了重要支撑。本书以两大前沿技术领域为核心,系统性地呈现了两部深度技术著作的精华:…...
