iOS中关于ARC的一些理解

什么是ARC

ARC(Automatic Reference Counting):自动引用计数,是苹果在2011年与iOS5一同推出的一项 编译器特性 。与MRC相似,只是由编译器在编译时,自动的在代码中合适的地方插入retain,release和autoreleas,且插入的代码不可见(事实上编译器并不是通过发送retain等消息来插入的,而是调用类似objc_retain这样的函数来完成同样的工作,所以ARC中,实际上是没有retain这样的系统消息的)。

引用计数

在Objective-C中,对于每个Objective-C对象,都有一个引用计数器来控制自身应该在何时被回收。当引用计数器中的计数为0时,对象被销毁,同时内存被回收。

ARC中,我们不能再显式的向对象发送retain或release消息,而是通过对应的关键字修饰指针,使其指向的对象能被编译器正确处理。只要指针没有置空,对象就一直保持在堆上。

__strong:强指针,默认指针类型。被strong关键字修饰的指针,指向的对象会同时被指针持有,即引用计数+1。在任何情况下,只要还有strong指针指向一个对象,此对象就不会被销毁。

__weak:弱指针,被weak关键字修饰的指针,指向的对象不会被指针持有,即对象的引用计数不变(因此,若没有一个强指针指向该对象,会出现弱指针指向的对象被创建的同时被销毁)。当没有任何一个强指针指向同一对象时,对象销毁,weak指针自动的被置为nil。

__unsafe__unretained:弱指针,不被编译器管理的内存对象。当对象被销毁时,指针不会被自动置为nil,指针依然指向目标地址,此时如果使用该指针,会出现BAD_ACCESS错误。

__autorelease:被__autorelease关键字修饰的指针,等同于MRC中接收autorelease消息。

对于属性的修饰关键字:

strong:等同于__strong;

copy:如果赋值来源是不可变对象,copy可以认为等同于strong,目标对象的引用计数+1。如果赋值来源是可变对象,copy修饰的属性会对目标对象执行深拷贝动作,在一块新的地址上存储一个相同的值,并指向新开辟的这块地址。copy得到的对象永远是不可变对象。

weak:等同于__weak;

assign:等同于__unsafe__unretained。

在ARC中,可以认为weak和assign,区别仅在于修饰的指针变量,在对象销毁时,是否会被主动置为nil。

面试题目:weak是怎么在对象被销毁后,主动将修饰的指针置为nil的?

runtime维护了一个weak表,对于注册为weak的对象,系统会以weak指针指向的对象内存地址作为key,将之放入到一个hash表之中,表中的Value是weak指针地址的数组。当此对象的引用计数为0时会调用dealloc,之后首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除。

关于具体的实现方式,可以去看看 samwei12's 的blog

你可能感兴趣的:(iOS中关于ARC的一些理解)