//当该类型的对象引用计数为0时,系统会自动调用该类的dealloc方法来回收空间,该方法是由系统自动调用的,不能手动调用
//验证对象空间有没有回收,只要查看该类的dealloc方法有没有执行即可。</span>
[per2 release];//3 - 2
[per1 release];//2 - 1
NSLog(@"%lu", [per2 retainCount]);
[per2 release];//1 - 0,到这里系统已经将空间回收
//过度释放
//现象:当写完和内存引用计数-1有关的操作后,程序立即crash
//原因:空间被系统回收后,不能再做和引用计数-1有关的操作,否则会立即crash
//解决方案:删除
[per1 release];//过度释放
per2 = nil;
per1 = nil;
当引用计数为0时,系统会自动回收内存,我们只管理引用计数
野指针异常:
现象:可能会崩溃,也可能不会崩溃,写到某一行代码时,突然崩溃。(没有写任何和引用计数相关的代码)
产生原因:该对象的空间已经被回收,不能在访问没有所有权的对象.
解决方案:空间被系统回收之后(引用计数为0),禁止访问
NSLog(@"%lu", [per1 retainCount]);//这样写是野指针问题,因为系统已经回收该空间(引用计数已经为0),系统已经标记删除,由系统管理
// NSLog(@"%lu", [per1 retainCount]);
</span>
//alloc和dealloc对应(开辟和回收空间),retain和release对应(引用计数加1和减1)
@autoreleasepool {
Person *per = [[Person alloc] init];//0 - 1
NSLog(@"%lu", [per retainCount]);
//[per retain];//由1 - 2,如果加上这个代码,则会运行正确
//autorelease 会将声明为autorelease的对象放入离它最近的自动释放池中,当自动释放池销毁时,会向池中的每一个对象发送一个release消息。
@autoreleasepool {
[per autorelease];//1 - 0
[per autorelease];//第二次是过度释放问题
}
//[per autorelease];//1 - 0
NSLog(@"test");
}
//内存管理基本原则:
//如果你对一个对象进行了alloc,retain,copy之后,你就拥有了该对象的所有权,就必须对他进行release或autorelease(最好紧挨着)
//面试题
*/
//内存泄漏问题(没有立即释放空间,占用内存太多)<pre name="code" class="objc"> @autoreleasepool {
for (long i = 0; i < 10000000000; i++) {
Person *per = [[Person alloc] init];
[per autorelease];
}
}
//改为
@autoreleasepool {
for (long i = 0; i < 10000000000; i++) {
//@autoreleasepool {
Person *per = [[Person alloc] init];
[per autorelease];
}
}
}
//面试题
@autoreleasepool {
NSString *per = [[NSString alloc] init];//0 - 1
[per retain];//1 - 2
[per retain];//2 - 3
//第一个问题: 内存泄露
per = @"aa";//指针重指向, 又指向了常量区的aa
[per release];
[per release];
[per release];
//第二个问题: 常量区的内存由系统管理
}