内存管理的方式
垃圾回收(gc)
ios⽀持两种内存管理⽅式:ARC和MRC。而垃圾回收在电脑上还是有的。
MRC
Manual Reference Count,⼈⼯引⽤计数:内存的开辟和释放都由程序代码进⾏控制。相对垃圾回收来说,对内存的控制更加灵活,可以在需要释放的时候及时释放,对程序员的要求较⾼,程序员要熟悉内存管理的机制。
ARC
Auto Reference Count,⾃动引⽤计数:iOS 5.0的编译器特性,它允许⽤户只开辟空间,不⽤去释放空间。它不是垃圾回收!它的本质还是MRC,只是编译器帮程序员默认加了释放的代码。
内存管理机制
OC采⽤引⽤计数机制管理内存,当⼀个新的引⽤指向对象时,引⽤计数器就递增,当去掉⼀个引⽤时,引⽤计数就递减。当引⽤计数到零时,该对象就将释放占有的资源。
影响引⽤计数的⽅法
alloc: 开辟空间,引用计数 从 0 变成 1
Person *p1 = [[Person alloc]initWithName:@"aaa" Age:22]; NSLog(@"%ld", p1.retainCount);
retain:使引用计数 + 1
[p1 retain];
release:影响引用计数 - 1
当对象的引用计数为0的时候,该对象就会被系统释放。就不要去访问这个空间了,一访问就是野指针
[p1 release];
autorelease:需要自动释放池 去 - 1
在未来的某一时刻会自动 - 1,是依托自动释放池去释放的。出了自动释放池 相当于系统会给释放池中间调用了autorelease对象发送一个release消息。Person *p1 = [[Person alloc]initWithName:@"aaa" Age:21]; NSAutoreleasePool *pool = [[NSAutoreleasePool alloc]init]; [p1 autorelease]; NSLog(@"%ld", p1.retainCount); [pool release]; NSLog(@"%ld", p1.retainCount);
或者:
// 自动释放池 @autoreleasepool { [p1 autorelease]; }
dealloc方法
-dealloc是继承⾃⽗类的⽅法,当对象引⽤计数为0的时候,由对象⾃动调⽤。可以在自定义类中重写dealloc方法。
书写规范:一定要写在类的最上面
- (void)dealloc{
// 如果你要在dealloc中使用对象的话,一定要写在[super dealloc]的上面,避免出现野指针问题
NSLog(@"释放了");
// 如果重写dealloc方法 必须要调用父类的方法,保证对象可以被释放
[super dealloc];
}
内存管理的基本原则
引⽤计数的增加和减少相等,当引⽤计数降为0之后,不应该再使⽤这块内存空间。凡是使⽤了alloc、retain或者copy让内存的引⽤计数增加了,就需要使⽤release或者autorelease让内存的引⽤计数减少。在⼀段代码内,增加和减少的次数要相等。
copy
伪拷贝
拷贝完还是一个对象(直接返回)
- (id)copyWithZone:(NSZone *)zone
{
return nil;
}
浅拷贝
拷贝后 有2个对象,但是他们的值是同一个值。
引用计数变化:被拷贝的对象不变,拷贝出来的新对象从 0 - 1
- (id)copyWithZone:(NSZone *)zone
{
// 浅拷贝
Person *p = [[Person alloc]initWithName:_name Age:_age];
return p;
}
深拷贝
深拷贝:拷贝出新对象,并且对象的值也重新拷贝一份再赋值。
对字符串进行拷贝,拷贝的结果要看字符串这个类如何实现的拷贝方法,对不可变字符串的拷贝,其实相当于直接retain一次。可变字符串拷贝时就是真拷贝了一个新的出来。
- (id)copyWithZone:(NSZone *)zone
{
NSString *name = [_name copy];
Person *p = [[Person alloc]initWithName:name Age:_age];
return p;
}