首先来看一个参数:@property(详见:
@property在内存管理中的作用:
1、可以控制set方法的内存管理;
1> retain:release旧值,retain1新值(用于OC对象);
2> assign:直接赋值,不做任何内存管理(默认,用于非OC对象类型);
3> copy:release旧值,retain新值(一般用于NSString *)。
2、控制需不需要生成set方法;
1> readwrite:同时生成set和get方法(默认);
2> readonly:只生成get方法;
3、多线程管理;
1> atomic:性能低(默认);
2> nonatomic:性能高
关于内存管理中的循环retain问题的解决:
循环retain场景:比如A对象retain了B对象,B对象retain了A对象;
循环retain的弊端:这样会导致A对象和B对象永远无法销毁(因为互相引用);
循环retain的解决方法:两端相互引用时应该一端用retain,一端用assign。
以上基本就是MRC机制下的内存管理方法,下面介绍ARC机制下的内存管理方法。首先,要了解autorelease。
autorelease方法的基本使用:
1> 给对象发送一条autorelease消息,会将对象放到一个自动释放池(autoreleasepool)中;
2> 当自动释放池被销毁时,会对池子里面的所有对象做一次release操作;
3> 会返回对象本身;
4> 调用完autorelease方法后,对象的计数器不变。
autorelease的好处:
1> 不用在关心对象释放的时间;
2> 不用在关心什么时候调用release。
autorelease的使用注意:
1> 占用内存较大的对象不要随便使用autorelease;
2> 占用内存较小的对象使用autorelease,没有太大影响。
自动释放池(autoreleasepool):
在iOS程序运行过程中,会创建无数个池子,这些池子都是以栈结构存在(先进后出);当一个对象调用autorelease方法时,会将这个对象放到栈顶得释放池。
自动释放池的创建方式:
1> iOS 5.0前:
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
[pool release];
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
[pool release];
2> iOS 5.0后:
@autoreleasepool {
}
@autoreleasepool {
}
autorelease的常见错误:
1、alloc之后调用了autorelease,又调用了release;
Person *p = [[[Person alloc] init] autorelease];
[p release];
Person *p = [[[Person alloc] init] autorelease];
[p release];
2、连续调用多次autorelease。
Person *p = [[[[Person alloc] init] autorelease] autorelease];
Person *p = [[[[Person alloc] init] autorelease] autorelease];
autorelease和release使用对比
/* 使用release */
Book *book = [[Book alloc] init];
[book release];
/* 使用autorelease */
Book *book = [[[Book alloc] init] autorelease];
// 不要再调用[book release];
/* 使用release */
Book *book = [[Book alloc] init];
[book release];
/* 使用autorelease */
Book *book = [[[Book alloc] init] autorelease];
// 不要再调用[book release];
autorelease的使用场合:
一般可以为类添加一个快捷创建对象的方法:
+ (instancetype)book {
return [[[self alloc] init] autorelease];
}
+ (instancetype)book {
return [[[self alloc] init] autorelease];
}
Book book]就可以获得和使用新建的Book对象,根本不用考虑在什么时候释放Book对象。
一般来说,除了alloc、new或copy之外的方法创建的对象都被声明了autorelease。
比如下面的对象都已经是autorelease的,不需要再release。
NSNumber *number = [NSNumber numberWithInt:250];
NSString *name = [NSString stringWithFormat:@"jack"];
NSString *stu = @"rose";
NSNumber *number = [NSNumber numberWithInt:250];
NSString *name = [NSString stringWithFormat:@"jack"];
NSString *stu = @"rose";
ARC是自iOS 5.0之后增加的新特性,是iOS程序员的福音。
ARC的全称是:Automatic Reference Counting(自动引用计数)
MRC的全称是:Manual Reference Counting(手动引用计数)
ARC的实现细节:
编译器会自动在适当地地方插入适当的retain、release、autorelease等语句;也就是说,编译器会自动生成内存管理的代码,不用程序员手动编写。
ARC注意点:
1、ARC是编译器特性,而不是运行时特性;
2、ARC不是其他语言中的垃圾回收,有着本质的区别。
ARC的优点:
1、完全消除了手动管理内存的繁琐,让程序员更加专注于app业务;
2、基本上能够避免内存泄漏;
3、有时还能更加快速,因为编译器还可以执行某些优化。
只要还有一个强指针变量指向对象,对象就会保持在内存中。
强指针:默认所有的指针变量都是强指针;被_strong修饰的指针。
弱指针:被_weak修饰的指针。
ARC使用细节:
不能调用release、retain、autorelease、retainCount
可以重写dealloc,但是不能调用[super dealloc];
ARC中的@property:
strong:用于OC对象,相当于MRC中的retain。
weak:用于OC对象,相当于MRC中得assign。
assign:用于基本数据类型,跟MRC中得assign一样。
copy: 一般用于NSString,跟MRC中得copy一样。