内存问题
- 野指针异常:访问所有权的内存,如果想要安全访问,必须确保空间还在(确保访问的内存不是僵尸对象)
- 内存泄露:空间使用完之后没有及时释放
- 过度释放:对一块空间释放多次,立刻crash
- 内存溢出:所有存储空间被占用
管理内存的三种方式
- 垃圾回收机制:程序员只要开辟存储空间,系统会自动回收内存。Java采用的机制。
- MRC:手动引用计数机制,由开发员开辟空间,手动添加影响引用计数增加或减少的方法,能够灵活的控制空间合适释放
- ARC:自动引用计数机制,是iOS 5.0 推出的,基于MRC,不需要程序员手动添加管理内存代码,编译器会在合适的地方添加管理内存的代码
引用计数机制:
iOS采用计数器来管理内存,当你拥有这个对象的时候,需要使用该对象的引用计数+1,当对象的引用计数器为0的时候,表示没有任何对象对该对象持有,那么这个时候系统会
自动调用dealloc方法来回收对象的存储空间
影响引用计数器的方法
使引用计数器+1的方法: alloc, retain, copy
使引用计数器-1的方法:release, autoreleae
retainCount 是ARC才有的机制所以要将MRC转换ARC
如图所示:将ARC模式关闭
1.release 使用引用计数减一
Person *person = [[Person alloc]init];
NSLog(@"retainCount = %lu",person.retainCount);
[person release];
NSLog(@"retainCount = %lu",person.retainCount);
2. retain使引用计数加一
Person *person = [[Person alloc]init];
NSLog(@"retainCount = %lu",person.retainCount);
[person retain];
NSLog(@"retainCount = %lu",person.retainCount);
3.野指针和僵尸对象
Person *person = [[Person alloc]init];
NSLog(@"retainCount = %lu",person.retainCount);
[person release]; //
NSLog(@"%lu",person.retainCount); // [person release]; //0 当对象的引用计数为0,系统自动调用dealloc方法 person = nil; // 对象置为nil,防止野指针异常
// 僵尸对象指的是:对象的引用计数器变成0后,该块地址被回收,变成不可访问的内存。
// 野指针指的是:指向僵尸对象(不可用内存)的指针,给野指针发送消息会报错(EXC_BAD_ACCESS)
// 空指针:没有指向任何东西的指针(存储的东西是nil、NULL、0),给空指针发送消息不会报错
4.开启 zombie 检测
5. 自动释放池
// 定义一个自动释放池
// 第一种形式
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; // pool 1 Person *p1 = [[Person alloc] init];
NSLog(@"%lu", p1.retainCount); [p1 retain]; //
[p1 autorelease];
NSLog(@"p1 = %lu", p1.retainCount);
NSLog(@"pool = %lu", pool.retainCount);
[pool release];
NSLog(@"p1 = %lu", p1.retainCount); // 1
// 对象使用 autorelease是在未来的某一时刻让对象的引用计数器减一,这个某一时刻是指碰到自动释放池子之后才会释放
// iOS 5.0 之后推荐使用的
// 第二种形式 @autoreleasepool { Person *p2 = [[Person alloc]init];
// [p2 retain];
p2.name = @"p2";
[p2 autorelease];
NSLog(@"p2 = %lu", p2.retainCount); Person *p3 = [[Person alloc]init];
// [p2 retain];
p3.name = @"p3";
[p3 autorelease];
NSLog(@"%lu", p3.retainCount);
}
// 总结:自动释放池的作用:自动释放池会在销毁的时候检查内部有没有autorelease对象,如果有autorelease对象,让该对象的引用计数做一次减一操作(即让池子内所有的引用对象减一)
// 销毁方式 以栈的形式销毁的
6. 内存管理原则
// 凡是使用alloc, retain让对象的引用计数加一,相应的就该使用release或者autorelease让对象的引用计数减一,业绩也是说增加的次数要和减少的次数相等,才能保障对象的引用计数始终唯0,对象才可以被销毁
// 不想等的情况下,会出现
// 1. 内存泄露:增加的次数大于减少的次数
// 情况一
Person *person1 = [[Person alloc]init];
// 情况二
Person *person2 = [[Person alloc]init];
[person2 retain]; // 2
[person2 release]; // 1
// 情况三
Person *person3 = [[Person alloc]init];
person3 = nil;
[person3 release];
// 2.过度释放:增加的次数小于减少的次数
Person *person1 = [[Person alloc]init]; // 1
[person1 retain]; // 2
[person1 release]; // 1
[person1 release]; // 0 [person1 release]; // 过度释放
// 3.野指针异常: 增加的次数等于减少的次数,还继续访问
Person *person =[[Person alloc]init];
[person retain];
[person release];
[person release];
person = nil;
NSLog(@"person = %lu",person.retainCount);