ARC是编译器(时)特性,而不是运行时特性,更不是垃圾回收器(GC)。
OC是使用引用计数机制(retainCount)来管理内存。内存每被引用一次,该内存的引用计数+1,每被释放一次引 用计数-1。当引用计数= 0 的时候,调用该对象的 dealloc 方法,来彻底从内存中删除该对象。
alloc,allocWithZone,new(带初始化)时:该对象引用计数 +1;
retain:手动为该对象引用计数 +1;(使用set方法赋值时,实质上是会先保留新值,再释放旧值,再设置新值,避免新旧值一样时导致对象被释放的的问题)
copy:对象引用计数 +1;(使用set方法赋值时,实质上是会先拷贝新值,再释放旧值,再设置新值)
mutableCopy:生成一个新对象,新对象引用计数为 1;
release:手动为该对象引用计数 -1;
autorelease:把该对象放入自动释放池,当自动释放池释放时,其内的对象引用计数 -1;(ARC下可以通过__autoreleasing修饰符进行标记,否则的话看方法名, 非alloc/new/copy/mutableCopy开头的方法编译器都会自动帮我们调用autorelease方法.)
ARC 下涉及到的 一些 修饰符
1.关键字 __strong 默认值
* 在程序中定义的引用,默认就是强引用,所谓的强引用指向一个对象时,对象的引用计数器会自动加1,当引用超出作用域,对象的引用计数器自动减1,
* 定义强引用:__Strong Student* stu = [[Student alloc] init];
* 使用set方法赋值时,实质上是会先保留新值,再释放旧值,再设置新值,避免新旧值一样时导致对象被释放的的问题。
* 当一个对象被引用指向时,此对象会隐式的retain一次,当强引用超出作用域时,指向的对象会隐式的release一次
* 引用在使用的时候,会根据作用域的范围,自动做加1减1操作
2.关键字__weak 弱引用
__weak Student* stu = [[Student alloc] init];
* 使用set方法赋值时,实质上不保留新值,也不释放旧值,只设置新值。
* 仅仅就是指向对象,
* 当一个弱引用指向的对象被销毁时,弱引用本身会自动的赋值为nil
3.关键字 __autoreleasing 用于标示自动释放的对象(变量)
* 在对象(变量)被标记之后,ARC会将变量放到自动释放池中(@autoreleasepool)等自动释放池销毁时(自动释放池的销毁时间是确定的,一般是在程序事件处理之后释放,或者由我们自己手动释放),会给标记的对象发送release消息,将其引用计数减1(如果自动释放池向对象发送release消息后对象的引用计数仍大于1,对象就无法销毁),并回收内存
4.__unsafe_unretained 不安全的弱引用,若没有任何强引用指针指向该变量,不会自动设为空,会成为野指针(为了兼容iOS5以下版本的产物,可以理解成MRC下的weak,现在基本用不到)。
注:@autoreleasepool实质上是一个__AtAutoreleasePool的结构体对象;这个结构体会在初始化时调用objc_autoreleasePoolPush()方法,会在析构时调用objc_autoreleasePoolPop()方法(关键点:入栈,出栈,双向链表)。这个地方有详细解释
那么在内存管理中也是有一些规则的:
1)当你通过new、alloc或copy方法创建一个对象时,它的引用计数为1,当不再使用该对象时,应该向对象发送release或者autorelease消息释放对象()。
2)当你通过其他方法获得一个对象时,如果对象引用计数为1且被设置为autorelease,则不需要执行任何释放对象的操作;
3)如果你打算取得对象所有权,就需要保留对象并在操作完成之后释放,且必须保证retain和release的次数对等
那么ARC 实际上帮我们做了哪些事?
ARC不是垃圾回收,也并不是不需要内存管理了,它是隐式的内存管理,编译器在编译的时候会在代码插入合适的ratain和release语句,相当于在背后帮我们完成了内存管理的工作。
什么时候需要自己手动创建autorelease pool?
1)你写的程序不是基于UI framework, 例如命令行项目
2)你写的循环创建了大量临时对象 -> 你需要在循环体内创建一个autorelease pool block并且在每次循环结束之前处理那些autoreleased对象. 在循环中使用autorelease pool block可以降低内存峰值
3)你创建了一个新线程,当线程开始执行的时候你必须立马创建一个autorelease pool , 否则你的应用会造成内存泄露.