引用计数
iOS内存管理机制的核心是引用计数
概念
引用计数简单来说,就是管理应用程序创建的每一块内存空间的生命周期
当一个对象创建时,系统就为他分配了一块内存空间,它的引用计数从0增加到1,表示有一个对象拥有这块内存的所有权;
如果有另一个对象也指向这块内存空间,则它的引用计数加1变为2;
之后如果一个对象不再指向这块内存空间,则它的引用计数减1,当一块内存空间的引用计数变为0,表示没有对象拥有这块内存空间,系统便会释放掉这块内存空间。
分类
引用计数分为 ARC(自动引用计数)和 MRC(手动引用计数)
ARC的本质其实也是MRC,两者的区别在于ARC是由系统来管理程序的内存空间,在系统认为合适的时间和地点释放引用计数为0的内存空间,而MRC则是由开发者自己管理内存空间的分配和释放。
ARC写代码很方便,不但减少了代码量,而且降低了内存出错的概率,但因为ARC不一定会及时释放内存空间,所以程序有时会占用比较大的内存;而MRC通过手动管理,及时释放不需要的内存空间,就可以保证程序长时间处于良好的运行状态。
关键字
引用计数的关键字有:alloc、strong、retain、copy、release、autorelease,其中strong关键字只用在ARC,作用等同于retain。
alloc
创建新对象并分配内存空间,每次调用alloc都是开辟一块新的内存空间,引用计数从0增加到1;
retain
创建新对象指向原对象的内存空间,原对象的内存空间引用计数加1,不会开辟新内存空间;
例如
NSObject *obj2 = [obj1 retain];
表示obj2指向并拥有obj1指向的内存空间;
copy
复制一个对象到一块新的内存空间上,返回一个新的对象,新对象的内存空间引用计数从0增加到1,
而原对象内存空间的引用计数不会变化,虽然内容一样,但实际上是两块内存空间;
copy分类:浅拷贝、普通深拷贝、真正深拷贝
浅拷贝: 只拷贝地址,与retain等同;
普通深拷贝:拷贝内容,会开辟新的内存空间存放对象,与retain不一样;
真正深拷贝:是对于容器类说的,如数组、字典、集合(包括可变和不可变类型),与普通深拷贝不同的是,真正深拷贝除了开辟新的内存空间存放容器类对象,连同容器类的元素都进行了深拷贝,为每个元素都开辟了新的内存空间。
release
使对象的内存空间的引用计数减1,若引用计数为0,系统会立刻释放这块内存;
注意:如果对引用计数为0的对象再调用release,会导致内存崩溃。
autorelease
作用和release一样,区别在于不会立刻减1,相当于一个延迟的release;
autorelease会在程序离开自动释放池时执行,通常系统会自动生成自动释放池@autoreleasepool{...},也可以自己设定自动释放池如:
@autoreleasepool{
NSObject *obj1 = [NSObject alloc] init];
[obj1 autorelease];
return obj1;
}
当程序执行到"}"时,obj1的引用计数就会减1,一般用于方法返回值的释放。
MRC内存释放
当一个对象的引用计数为0,内存空间被释放时,会触发dealloc方法.
在MRC下定义类,必须在deallc方法中将关键字为retain、copy的属性release一次,避免造成内存泄漏,注意先调用[super dealloc];
-(void)dealloc{
[super dealloc];
[obj1 release];
[obj2 release];
}