【Objective-C内存管理之引用计数】
自动引用计数
- ARC(Automatic Reference Counting)是指内存管理中对引用采取自动计数的计数
在Objective-C中采用Automatic Reference Counting(ARC)机制,让编译器来进行内存管理。在新一代Apple LLVM编译器中设置ARC为有效状态,就无需再次键入retain或者release代码,这就降低程序崩溃、内存泄露等风险的同时,很大程度上减少了开发程序的工作量。编译器完全清除目标对象,并能立刻释放那些不再被使用的对象。如此一来,引用程序具有可预防性,且能流畅运行,速度也将大幅提升
- 在这之前学习过iOS属性关键字–retain引用计数的了解
现在编译器都是自动进行内存管理,今天了解了在这之前程序员是如何手动进行内存管理的
引用计数理解
自动引用计数的思考方式
- 自己生成的对象自己所持有
- 非自己生成的对象自己也能持有
- 不再需要自己持有的对象时释放
- 非自己持有的对象无法释放
- 注意生成,持有 ,释放,废弃的理解在下面解释
对象操作与OC方法的对应
对象操作 | Objective-C |
---|---|
生成并持有对象 | alloc/new/copy/mutableCopy 方法 |
持有对象 | retain方法 |
释放对象 | release方法 |
废弃对象 | dealloc方法 |
- 注意在生成并持有对象的四个方法中,alloc是NSObject类方法,其余三个都是实例方法
思考方式
自己生成的对象自己持有
- 使用
alloc new copy mutableCopy
开头名的方法生成的对象都是自己所持有的
id obj = [[NSObject alloc] init];
id obj1 = [NSObject new];
obj1 和 obj 的类型是完全一致的
- 根据上述的“使用以下名称开头的方法名”,下列名称也意味着自己生成并持有对象:遵循驼峰命名规则
allocMyObject
newThatObject
copyThis
mutableCopyYourObject
- 但是对于以下名称,即使用alloc/new/copy/mutableCopy 名称开头,却并不属于同一类别的方法(即就是加上后缀形成其他单词)。
allocate
newer
copying
mutableCopyed
非自己生成的对象自己也能持有
- 用alloc/new/copy/mutableCopy以外的方法取得的对象,因为非自己生成并持有,所以自己不是该对象的持有者,书上用了NSMutableArray类的array类方法
- 上网查询了ARC环境下获取引用计数的方法
CFGetRetainCount((__bridge CFTypeRef)(obj))
,系统已经自动进行内存管理
//获取非自己生成并持有的对象
id obj = [NSMutableArray array];
//获取到了对象,但是自己并不持有
//自己持有对象
[obj retain];
- 通过
retain
方法,非自己持有生成的对象跟用alloc/new/copy/mutableCopy
方法生成并持有的对象一样,成了自己所持有的对象。
不再需要自己持有的对象时释放
- 自己持有的对象,一旦不再需要,持有者有义务释放该对象。释放使用
release
方法。
//自己生成并持有对象
id obj = [[NSObject alloc] init];
//自己持有对象
[obj release];
//释放对象
//指向对象的指针仍然被保留在变量obj中,似乎可以访问,但对象一经释放绝不可访问,否则程序就会报错。
- 如此,用过alloc方法由自己生成并持有的对象就通过release方法释放了。自己生成而非自己所持有的对象,若用retain方法变为自己持有,也同样可以用release方法释放
//取得非自己生成并持有的对象
id obj = [NSMutableArray array];
//取得的对象存在,但自己不持有对象
[obj retain];
//自己持有对象
[obj release];
//释放对象
//对象不可再被访问
- 当需要用某个方法生成对象,并且产生非自己生成但持有的对象,方法如下
- (id) allocObject {
//自己生成并持有对象
id obj = [[NSObject alloc] init];
//自己持有对象
return obj;
}
//取得非自己生成并持有的对象
id objFirst = [objZero allocObject];
//自己持有对象
- 探究调用了方法自己生成并非自己持有的对象(NSMutableArray方法)
- (id) object {
id obj = [[NSObject alloc] init];
//自己持有对象
[obj autorelease];
//取得的对象存在,但自己不持有对象
return obj;
}
- 上面我们使用了autorelease方法,用该方法可以使取得的对象存在,但自己不持有对象。autorelease提供这样的功能
release 和 autorelease
- 使用autorelease方法,可以使取得的对象存在,但自己不持有对象。autorelease提供这样的功能,使对象在超出制定的生存范围时能够自动并正确的释放(调用release方法
- 使用NSMutableArray类的array类方法等可以取得谁都不持有的对象,这些方法都是通过autorelease而实现的。此外,根据上文的命名规则,这些用来取得谁都不持有的对象的方法名不能以alloc\ new\ copy \ mutableCopy 开头。
- retain方法将调用autorelease方法取得的对象变为自己持有
无法释放非自己持有的对象
- 对于用
alloc\ new\ copy \ mutableCopy
方法生成并持有的对象,或是用retain方法持有的对象,由于持有者是自己,所以在不需要该对象时需要将其释放,而由此之外所得到的对象绝对不能释放。倘若在应用程序中释放了非自己所持有的对象就会发生崩溃
自己持有的二次释放
// 自己生成并持有对象
id obj = [[NSObject alloc] init];
//自己持有对象
[obj release];
//对象已释放
[obj release];
//释放之后再次释放已非自己持有的对象,应用程序崩溃
//崩溃情况:再度废弃已经废弃了的对象时崩溃
非自己持有的释放
id objFirst = [objZero object];
//取得的对象存在,但自己不持有对象
[objFirst release];
//释放了非自己持有的对象,这肯定会导致应用程序崩溃。
总结
- 了解引用计数规则是学习iOS内存管理的开始,很重要!