@一、范围:任何继承了NSObject的对象,对基本数据类型无效。
@二、原理:
1.每个对象内部都保存一个与之相关联的整数,称为引用计数器
2.当使用alloc、new或者copy创建一个对象的时候,对象引用计数器被设置为1
3.给对象发送一条retain消息,可以使引用计数器值+1
4.给对象发送一条release消息,可以使引用计数器值-1
5.当一个对象的引用计数器值为0时,那么它将被销毁,其占用的内存被系统回收,OC也会自动向对象发送一条dealloc消息。一般会重写dealloc方法,在这里释放相关资源。一定不要直接调用dealloc方法。
6.可以给对象发送retainCount消息活得当前的引用计数器值。
@三、管理规则:
1.谁创建,谁释放(谁污染,谁治理)。如果你通过alloc,new或(mutable)或copy来创建一个对象的话,那么你必须调用release或者autorelease。换句话说,不是你创建的你就不用释放。
2.一般来说,除了alloc,new,或者copy之外的方法创建的对象都被声明了autorelease。
3.谁retain,谁release。只要你调用了retain,无论这个对象啊是如何生成的,你都要调用release。
@四、示例
@interface Student : NSObject
@property int age;
@end
上面是Student类的声明部分.h文件,下面是实现部分.m文件
@implementation Student
@synthesize age = _age; // 在xcode4.5环境下可以省略
- (void)dealloc {
NSLog(@"%@被销毁了", self);
[super dealloc];
// 一定要调用super的dealloc方法,而且最好放在最后面调用
}
@end
我们写一个测试方法来说明内存引用计数的变化
Student *stu = [[Student alloc] init]; // 1 // z代表无符号
NSLog(@"count:%zi", [stu retainCount]);
[stu retain]; // 2
NSLog(@"count:%zi", [stu retainCount]);
[stu release]; // 1
NSLog(@"count:%zi", [stu retainCount]);
[stu release]; // 0
// [stu release]; // 会发生野指针错误,也就是说访问了不属于你的内存
再写一个示例说明:
// Student对象的计数器永远为1,所以不可能被释放
[[Student alloc] init].age = 10;
[Student new].age = 10;
// 上面的代码都有内存泄露
@五、对象之间的内存管理
对象之间的内存管理是经常见到的情况,我们来模拟一下学生(Student)和书(Book)之间的内存管理。首先Student类声明和实现部分如下:
@interface Student : NSObject {
Book *_book;
}
@property int age;
- (id)initWithAge:(int)age;
@property Book *book;
- (void)readBook;
@implementation Student
#pragma mark - 生命周期方法
#pragma mark 构造方法
- (id)initWithAge:(int)age {
if ( self = [super init] ) {
_age = age;
}
return self;
}
#pragma mark 回收对象
- (void)dealloc {
// 释放Book对象
[_book release];
// [self.book release];
NSLog(@"student:%i 被销毁了", _age);
[super dealloc];
}
#pragma mark - getter和setter
// @synthesize book = _book;
// 如果自己手动实现了getter和setter,xcode就不会自动生成@synthesize
// 也就不会自动生成_book
// getter和setter的默认实现
- (void)setBook:(Book *)book {
if (_book != book) {
// 先释放旧的成员变量
[_book release];
// 再retain新传进来的对象
_book = [book retain];
}
}
- (Book *)book {
return _book;
}
#pragma mark - 公共方法
#pragma mark 读书
- (void)readBook {
NSLog(@"当前读的书是:%f", _book.price);
}
//#pragma mark - 私有方法
//#pragma mark 私有方法1
//- (void)test1 {}
//#pragma mark 私有方法2
//- (void)test2 {}
//#pragma mark 私有方法3
//- (void)test3 {}
@end
Book类的声明和实现部分如下:
@interface Book : NSObject
@property float price; // 价格
- (id)initWithPrice:(float)price;
@end
@implementation Book
- (id)initWithPrice:(float)price {
if ( self = [super init] ) {
_price = price;
}
return self;
}
- (void)dealloc {
NSLog(@"book:%f 被销毁了", _price);
[super dealloc];
}
@end
下面在main方法中对内存管理进行测试分析,示例如下:
void test(Student *stu) {
// book:1
Book *book = [[Book alloc] initWithPrice:3.5];
// book:2
stu.book = book;
// book:1
[book release];
// book:1
stu.book = book;
stu.book = book;
// book2:1
Book *book2 = [[Book alloc] initWithPrice:4.5];
// book2:2
stu.book = book2;
// book2:2
stu.book = book2;
// book2:1
[book2 release];
// book2:1
stu.book = book2;
}
void test1(Student *stu) {
[stu readBook];
}
int main(int argc, const char * argv[])
{
@autoreleasepool {
// stu:1
Student *stu = [[Student alloc] initWithAge:10];
// stu:1
// book:1
// book2:1
test(stu);
// stu:1
// book:1
// book2:1
test1(stu);
// stu:0
// book2:0
// book:1
[stu release];
// stu = nil; 清空stu这个指针,stu就会变成空指针
// [stu release]; // 野指针(会报错)
[nil release]; // 空指针(不会报错)
}
return 0;
}
通过分析,加深对内存管理的了解。熟记口诀,牢记原则。