【iOS】引用计数

引用计数

自动引用计数

自动引用计数(ARC,Automatic Reference Counting)是指内存管理中对引用采用自动计数的技术。对于ARC技术来说,最重要的就是***在LLVM编译器中设置ARC为有效状态,就不用再次输入retain和release代码***。

当满足下列条件时,就无须手动输入retain和release代码了:

  • 使用XCode4.2或以上版本
  • 使用LLVM编译器3.0或以上版本
  • 编译器选项中设置ARC为有效

引用计数

我们可以用一个房间内开关灯来说明引用计数这个问题,当房间内进入第一个人时开灯,最后一个人离开时关灯,在这个过程中,需要照明的人数就是引用数。

如上图所示,就是引用计数的原理

那么对于OC对象来说,下图即可展示两者的对应动作:

如此来说,OC中的内存管理即入下图所示:

内存管理的思考方式

在内存管理中,正确、客观的思考方式是:

  1. 自己生成的对象,自己所持有
  2. 非自己生成的对象,自己所持有
  3. 不再需要自己持有的对象时释放
  4. 非自己持有的对象无法释放

在上面的思考方式中多次出现"生成"、"持有"、"释放"三个词,在OC内存管理中还有"废弃"一词,下面展示各个词表示的OC方法:

操作对象 Objective-C方法
生成并持有对象 alloc/new/copy/mutableCopy等方法
持有对象 retain方法
释放对象 release方法
废弃对象 dealloc方法

这些有关 Objective-C 内存管理的方法,实际上不包括在该语言中,而是包含在 Cocoa 框架中用于 OS X、iOS 应用开发。Cocoa 框架中 Foundation 框架类库的 NSObject 类担负内存管理的职责。Objective-C内存管理中的 allocretain/release/dealloc 方法分别指代 NSObject 类的 alloc 类方法、retain 实例方法、release 实例方法和 dealloc 实例方法。


自己生成的对象,自己所持有

使用以下名称开头的方法名意味着自己生成的对象只有自己持有:

  • alloc
  • new
  • copy
  • mutableCopy

代码格式:

objective-c 复制代码
id obj = [[NSObject alloc] init];
id obj = [NSObject new];
//注意:上面两种生成并持有对象的方法是完全一致的

对于以下这些名称开头的方法名也意味着自己生成并持有对象:

  • allocMyObject
  • newThatObject
  • copyThis
  • mutableCopyYourObject

非自己生成的对象,自己也能持有

除了上述方法以外的方法生成的对象,因为不是自己生成并持有,所以自己不是该对象的持有者。下面举例说明

objective-c 复制代码
id obj = [NSMutableArray array];

[obj retain];
//通过retain方法,非自己生成的对象跟用alloc等方法生成的对象一样,成了自己所持有的。

不再需要自己持有的对象时释放

自己持有的对象,一旦不再需要时,持有者有义务释放该对象,使用release方法释放。

objective-c 复制代码
id obj = [[NSObject alloc] init];

[obj release];

这里举了一个例子说明,当使用retain方法变成自己持有的时候,也可以使用release释放。

无法释放非自己持有的对象

我们在使用alloc等方法或者retain方法将持有者变成自己时,需要在不需要时释放该对象,但是如果是非自己持有的对象,我们不能释放他,如果释放就会造成应用程序崩溃。例如自己生成对象后不需要时释放了两次,就会导致这个问题。

objective-c 复制代码
id obj = [[NSObject alloc] init];

[obj release];

[obj release];

这样做,由于第一次已经释放过了,再次释放就会导致应用程序崩溃。


autorelease和release的区别

当我们使用alloc等方法向实现取得的对象存在但是自己不持有就需要用到autorelease方法。下面举例说明:

objective-c 复制代码
id obj = [[NSObject alloc] init];

[obj autorelease];

使用autorelease方法就可以实现,令对象在超出指定的生存范围时能够自动并正确的释放掉,在我们上文中举例的array就是通过调用autorelease实现的,当然,也可以通过retain方法再将其变成自己持有的。

上图即展示了autorelease和release的区别。

相关推荐
请揣满RMB6 小时前
贪心算法介绍
c++·算法·ios·贪心算法
low神7 小时前
React Native中如何调用iOS的Face ID和Android的生物识别,react-native-biometrics
android·前端·javascript·react native·react.js·ios·前端面试题
面壁者LOGIC8 小时前
Invalid Executable The executable contains bitcode
ios·xcode
一只蜗牛儿8 小时前
自动驾驶综述 | 定位、感知、规划常见算法汇总
人工智能·macos·xcode
请叫我飞哥@20 小时前
Xcode 16 Pod init 报错
ide·macos·xcode
bugtraq20211 天前
Fyne ( go跨平台GUI )中文文档- 扩展Fyne (七)
开发语言·golang·xcode
菜的不敢吱声1 天前
假期学习笔记总结--iOS 自动释放池
笔记·学习·ios
骨子里的偏爱1 天前
Xcode报错:The request was denied by service delegate (SBMainWorkspace)
ide·macos·xcode
码农褚2 天前
uniApp 解决uniapp三方地图获取位置接口的请求次数限制问题,分别提供 Android 和 iOS 的实现方法(原生插件获取)
android·前端·ios·uni-app