当我们使用一个指针指向一块内存的时候,应该对这块内存做retain操作,引用计数+1。当我们不再使用这个指针指向这块内存,应该对这块内存做release操作,引用计数-1。这样可以使引用计数值一直保持等于指向这块内存的指针数量。retainCount返回当前内存的引用计数值。当内存被开辟后,默认的引用计数为1 。
Boy *boy =[[Boy alloc] init];
NSLog(@"%ld",boy.retainCount);
结果:2015-07-27 19:30:27.587 OC08,09_内存管理[427:8615] 1
Boy *boy =[[Boy alloc] init];
[boy retain];
NSLog(@"%ld",boy.retainCount);
2015-07-27 19:34:35.190 OC08,09_内存管理 [436:9846] 2
Boy *boy =[[Boy alloc] init];
[boy retain];
[boy release];
NSLog(@"%ld",boy.retainCount);
2015-07-27 19:36:35.985 OC08,09_内存管理[456:10672] 1
dealloc在.m文件中实现,dealloc要释放所有不能自己释放的成员变量
例子:
(1)在.h文件中声明三个属性:自定义初始化和便利构造器
@property(nonatomic , retain)NSMutableArray *arr;
@property(nonatomic, copy)NSString *name;
@property(nonatomic, assign)NSInteger age;
-(id)initWithName:(NSString *)name
age:(NSInteger)age;
+(Person *)personWithName:(NSString *)name
age:(NSInteger)age;
在.m文件中完成实现:
-(id)initWithName:(NSString *)name
age:(NSInteger)age{
self =[super init];
if (self) {
self.name=name;
self.age =age;
self.arr=[NSMutableArray array];
}
return self;
}
+(Person *)personWithName:(NSString *)name
age:(NSInteger)age
{
Person *person=[[Person alloc] initWithName:name age:age];
return [person autorelease];
}
在最后需要加一步dealloc:
-(void)dealloc
{
[_arr release];
[_name release];
[super dealloc];
}
实现整个类中的成员变量全部释放,在这里因为age是NSInteger类型,在栈区,系统会自动管理,不需要手动释放
上面这些是完整的声明和实现一个类内的成员变量的方法
NSString *str =@"1111";
NSLog(@"%ld",str.retainCount);
结果:2015-07-27 19:54:12.544 OC08,09_内存管理[495:16187] -1
release和autorelease的区别:release 马上会把对象的引用计数 -1,但是autorelease会延迟把对象的引用计数 -1
Boy *boy =[[Boy alloc] init];
[boy retain];
[boy retain];
NSLog(@"%ld",boy.retainCount);
[boy autorelease];
NSLog(@"%ld",boy.retainCount);
结果:
2015-07-27 20:06:16.356 OC08,09_内存管理[541:20376] 3
2015-07-27 20:06:16.357 OC08,09_内存管理[541:20376] 3
只要对象用autorelease 释放会把对象放入到系统的自动释放池中,等出了池子的范围,对象引用计数自动 -1,这个相当于java 的垃圾回收,对象释放由系统来管理
Boy *boy =[[Boy alloc] init];
[boy retain];
[boy retain];
NSLog(@"%ld",boy.retainCount);
@autoreleasepool {
[boy autorelease];
NSLog(@"%ld",boy.retainCount);
}
NSLog(@"%ld",boy.retainCount);
结果:
2015-07-27 20:19:02.918 OC08,09_内存管理[573:23991] 3
2015-07-27 20:19:02.919 OC08,09_内存管理[573:23991] 3
2015-07-27 20:19:02.919 OC08,09_内存管理[573:23991] 2
Boy *b =[Boy boy];
[b retain];
NSLog(@"%ld", b.retainCount);
[b release];
在这里 b 就有了对象的所有权,就可以对对象进行release的操作了
+(Boy *)boyWithName:(NSString *)name
hobby:(NSString *)hobby
{
Boy *boy=[[Boy alloc] initWithName:name hobby:hobby];
// 写便利构造器最后别忘了autorelease
return [boy autorelease];
}
(1).当对象放入到容器Array或字典中,对象会被容器进行一次持有,主要是为了防止空指针问题
(2).等对象从容器中移除掉之后,相应地会 -1
Boy *boy1 =[[Boy alloc] init];
[boy1 retain];
[boy1 retain];
NSLog(@"%ld",boy1.retainCount);
NSMutableArray *arr = [NSMutableArray arrayWithObjects:boy1, nil];
NSLog(@"%ld",[arr[0] retainCount]);
NSLog(@"%ld",boy1.retainCount);
[arr removeObjectAtIndex:0];
NSLog(@"%ld",boy1.retainCount);
结果:
2015-07-27 20:39:49.255 OC08,09_内存管理[608:31455] 3
2015-07-27 20:39:49.256 OC08,09_内存管理[608:31455] 4
2015-07-27 20:39:49.256 OC08,09_内存管理[608:31455] 4
2015-07-27 20:39:49.256 OC08,09_内存管理[608:31455] 3
(3).对象放到数组里,就可以直接释放对象,对象在数组中可以继续使用,直到数组消失时释放对象
Boy *b=[[Boy alloc] init];
NSArray *arr =@[b];
[b release];
版权声明:本文为博主原创文章,未经博主允许不得转载。
原文地址:http://blog.csdn.net/mltianya/article/details/47089421