一、内存管理的思考方式
内存管理的思想包括以下四点:
1、自己生成的对象,自己所持有
2、非自己生成的对象,自己也能持有
3、不再需要自己持有的对象时,要释放
4、非自己持有的对象 不能释放
自己生成的对象自己持有
alloc 、new、copy、mutableCopy
这些名称开头的方法名意味着自己生成的对象只有自己持有;上文中出现了一系列的自己这个词,可以将自己理解为“对象的使用环境”,也可以理解为编程人员“自身”。下面写出了自己生成并持有对象的源代码。我们在这里使用alloc方法:
//自己生成对象并持有对象
id obj =[ [NSObject alloc] init];
//自己持有对象
使用alloc类方法就能自己生成并持有对象。指向生成并持有对象的指针被赋给变量obj,另外,使用new类方法也可以生成并持有对象。和alloc时完全一致的
//自己生成对象并持有对象
id obj =[NSObject new];
//自己持有对象
copy方法利用基于NSCopying方法约定,由各类实现的copyWithZone:方法生成并持有对象副本。与copy方法类似,mutableCopy方法利用基于NSMutableCopying方法约定,由各类实现的mutableCopyWithZone:方法生成并持有对象的副本。 用这些方法生成的对象,虽然是对象的副本,但是和alloc、new一样,在”自己生成并持有对象“这点上没有发生改变;
非自己生成的对象,自己也能持有
用alloc/new/copy/mutableCopy以外的方法取得的对象,因为非自己生成并持有,所以自己不是该对象的持有者。看以下源代码:
//取得非自己生成并持有的对象
id obj = [NSMutableArray array];
//取得对象的存在,但自己不持有对象
上述代码中,NSmutableArray类对象被赋值给变量obj,但变量obj自己并不持有该对象。使用retain方法可以持有对象
//取得非自己生成并持有的对象
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;
}
如上,原封不动返回用alloc方法生成并持有的对象,就能让调用方也持有该对象。
//取得非自己生成并持有的对象
id obj1 = [obj0 allocObject];
//自己持有对象
那么,如何像调用[NSMutableArray array] 方法使取得的对象存在,但自己不持有对象,又是如何实现的呢?如下:
- ( id ) allocObject
{
//自己生成并持有对象
id obj = [[NSObject alloc ] init ];
[obj autorelease];
//取得对象的存在,但是自己不持有对象。
return obj;
}
上述代码中,我使用了autorelease方法。用该方法,可以使取得的对象存在,但自己不持有对象。以后有时间我会详细介绍autorelease。
id obj1 = [obj0 object];
//取得的对象存在,但自己不持有对象。
当然也可以像前文中的NSMutableArray一样,通过retain 方法,将调用autorelease方法取得的对象变为自己持有。
无法释放非自己持有的对象
对于由alloc/new/copy/mutableCopy 方法生成并持有的对象,或是retain方法持有的对象,由于持有者是自己,所以在不需要该对象的时候,需要将其释放。而由此以外所得到的对象绝对不能释放。如果在应用程序中释放了非自己所持有的对象就会造成崩溃。例如自己生成的并持有对象后,在释放完不再需要的对象之后再次释放。
//自己生成并持有对象
id obj = [[NSObject alloc] init];
//自己持有对象
[objc release];
//对象已被释放
[obj release];
//释放之后再次释放已非自己持有的对象,应用程序崩溃
//崩溃情况:再度废弃一经废弃了的对象时崩溃 、 访问一经废弃的对象时崩溃;
或者在“取得的对象存在,但自己不持有对象”时释放
id obj1 = [obj0 object];
[obj1 release];
//释放了非自己持有的对象,这肯定会导致应用程序崩溃
如这些例子所示,释放非自己持有的对象会造成程序崩溃。因此绝对不要去释放非自己持有的对象。
网友评论