前言
自动引用计数 是指内存管理中对引用采取自动计数的技术。
OC中ARC
模式是让编译器来替我们实现内存管理,我们无需再像在MRC
模式下需要手动键入retain
、release
来进行内存管理,不但降低了程序的出错概率也减少了程序的书写工作量。这里要提一下的是,拥有ARC
这样强大内存管理模式主要依赖于3.0
及以上版本的LLVM
编译器。
内存管理
OC内存管理的核心就是引用计数。我们以遛狗的场景为例来解释引用计数机制:
当人物A要出门遛狗的时候,A会在狗的脖子上栓一条链子牵着它。然后人物B也想同时溜这条狗(当然在现实中这种情况比较奇葩了,我们先不考虑这个),B会在狗的脖子上再拴上一条链子,这时两个人都对这条狗持有。当A不想溜了就会解开链子,此时狗的脖子上只有B持有的一条链子。当B也不想溜了,B也解开了狗脖子上的链子。此时狗的脖子上没有链子的束缚,狗就逃掉了。
上边情景中狗就是对象,人物在狗的脖子上拴上链子就是引用,解开狗脖子上的链子就是释放,狗脖子上链子数就是引用计数,当狗脖子上链子数为0狗会逃走,也就是当对象引用计数为0对象就被完全释放。
下面用一个图示解释一下:
对象的引用计数我们可以通过
[object retainCount];
获取。
图示中有四个动词:生成、持有、释放、废弃,这四个词分别对应OC中的哪些方法以及含义呢?
- 生成并持有:
alloc
/new
/copy
/mutableCopy
等
自己(这里的自己应该理解为生成对象的‘当前环境’)生成的对象,并持有它。 - 持有:
retain
非自己生成的对象,自己去持有它。 - 释放:
release
不再需要自己持有的对象,释放它。 - 废弃:
dealloc
对象不再被需要,被销毁。
其实这些内存管理的方法并不是OC语言本身提供的,而是NSObject
类本身肩负了内存管理的职责,alloc
是NSObject
类的类方法,retain
、release
、dealloc
是NSObject
类的实例方法。
生成并持有
除了使用alloc
、new
、copy
、mutableCopy
方法外:
使用alloc
、new
、copy
、mutableCopy
开头的方法同样意味着自己生成并持有,如:
allocOneObject
newThatObject
copyThis
mutableCopyYourObject
...
例如:
- (id)allocObject{
id object = [[NSObject alloc] init];
return object;
}
命名规则必须遵循驼峰写法,以下命名的方法则不属于自己生成并持有的范畴
alloconeobject
newthatobject
...
copy
、mutableCopy
方法是分别依赖于NSCopying
、NSMutableCopy
协议,由各类实现的copyWithZone:
、mutableCopyWithZone:
方法生成并持有的对象的副本。这些方法虽然生成的是副本,但依然是自己生成并持有。
持有
除了以上的方法外,我们来看看NSMutableArray
的array
方法
// 取得非自己生成的对象
id object = [NSMutableArray array];
// 通过retain函数,实现自己持有
[object retain];
释放
当我们不在需要持有对象的时候,我们就要释放对对象的持有,那就要用到release
方法
id object = [[NSMutableArray alloc] init];
// 释放持有
[object release];
id object = [NSMutableArray array];
[object retain];
// 释放持有
[object release];
都可以使用release函数进行释放
注意:我们是无法释放非自己持有的对象的,比如:
id object = [[NSObject alloc] init];
// 释放对象 自己不再持有
[object release];
// 当再次释放自己已经不再持有的对象就会报错
[object release]; // 所以这一句代码是错误的
ARC规则
上面介绍的大多数方法都是MRC
模式下我们手动管理内存的操作方式,在ARC
模式下以上的操作基本上都被在编译阶段自动处理了。
从本质上来讲ARC
模式只不过是MRC
的升级,内存管理的机制原理是没有变的。那么ARC
是怎样实现的呢?我们先要看一看ARC
中追加的所有权声明。
OC中为了处理对象,将变量类型定义为id
类型或各种对象类型,当ARC
模式时,id类型和对象类型必须附加所有权修饰符。所有权修饰符有以下4种:
- __strong
- __weak
- __unsafe_unretained
- __autoreleasing
__strong修饰符
__strong修饰符是id类型和对象类型的默认所有权修饰符,也就是以下代码
id object = [[NSObject alloc] init];
实际被附加了所有权修饰符
id __strong object = [[NSObject alloc] init];
版权声明:出自MajorLMJ技术博客的原创作品 ,转载时必须注明出处及相应链接!