先说下我们iOS程序员需要了解的内存四大区域:地址由低到高顺序为代码区,数据段,堆区,栈区
代码区:我们所写的代码编译成二进制就是存储在这个区域
数据段:存放全局变量、静态变量
堆区:通过malloc初始化的对象存储在堆区
栈区:函数中的局部变量都存储在栈区
MRC:手动内存管理
先看下我们平时在MRC下是如何手动管理内存的
#import "WPPerson.h"
int main(int argc,const char* argv[]) {
@autoreleasepool {
WPPerson*person = [[WPPerson alloc] init];
NSLog(@"--%@--",person);
[person release];
}
return0;
}
可以看到,当我们初始化一个对象,不用它的时候需要调用release进行释放,有时会出现提前释放了,这时可以在初始化对象时调用autoreleasefangfa ,这样编译器就会在恰当的时机调用release进行释放
WPPerson*person = [[[WPPerson alloc] init] autorelease];
当出现更复杂的情况,WPPerson对象内部使用到其他对象WPCat,看代码
WPCat有一个对外的方法eat
#import "WPCat.h"
@implementation WPCat
-(void)eat{
NSLog(@"[WPCat eat]");
}
-(void)dealloc{
NSLog(@"[WPCat dealloc]");
[super dealloc];
}
@end
WPPerson有一个成员变量WPCat,并对外提供了setter、getter方法
#import "WPPerson.h"
#import "WPCat.h"
@interface WPPerson()
{
WPCat*_cat;
}
@end
@implementation WPPerson
-(WPCat*)cat{
return_cat;
}
-(void)setCat:(WPCat*)cat{
_cat= cat;
}
-(void)dealloc{
NSLog(@"[WPPerson dealloc]");
[super dealloc];
}
@end
mian文件中
#import <Foundation/Foundation.h>
#import "WPPerson.h"
#import "WPCat.h"
int main(int argc,const char* argv[]) {
@autoreleasepool {
WPPerson*person = [[WPPerson alloc] init];
WPCat*cat = [[WPCat alloc] init];
[person setCat:cat];
[cat release];
NSLog(@"%@",[person cat]);
sleep(2);
[[person cat] eat];
[person release];
}
return0;
}
初始化WPPerson和WPCat对象,并将cat赋值给person的成员变量cat,person里面的setter方法内部只是简单的赋值操作,而且当赋值操作完成后释放之前的cat对象,这时如果person使用成员变量cat进行操作时会出现EXC_BAD_ACCESS坏内存访问错误,这里睡两秒只是为了体现这个问题,因为当cat调用release方法时,cat对象有可能没有马上被释放,而是加入到了自动释放池中,这时eat方法还是可以调用的,这样就导致cat对象被提前释放了,所以我们一般都会在setter方法中对cat对象进行retain操作,使cat对象的引用计数+1,这样cat对象就不会被销毁
-(void)setCat:(WPCat*)cat{
_cat= [cat retain];
}
但是这样又会导致cat对象不会被释放,造成内存泄漏,这时只需要在person的dealloc中将成员变量_cat
释放掉即可
-(void)dealloc{
[_cat release];
_cat=nil;
NSLog(@"[WPPerson dealloc]");
[super dealloc];
}
但如果中途person对象将成员变量重新赋值了一个cat对象,这样就会导致之前的cat对象没有被释放
WPPerson*person = [[WPPerson alloc]i nit];
WPCat*cat = [[WPCat alloc] init];
WPCat*cat1 = [[WPCat alloc] init];
[person setCat:cat];
[person setCat:cat1];
[cat release];
[cat1 release];
NSLog(@"%@",[person cat]);
sleep(2);
[[person cat] eat];
[person release];
这时只会有一个cat对象被释放,也就是最后赋值的cat1对象,而cat对象没有被释放,这时可以改进setter方法
-(void)setCat:(WPCat*)cat{
if(_cat!= cat) {
[_cat release];
_cat= [cat retain];
}
}
这就是我们在MRC环境下看到的setter方法写法的由来
网友评论