ARC是否确实保留了对对象的无主引用计数?
那么,如果某个对象的强引用计数达到0而该对象的未拥有引用计数大于0,则该对象将被取消初始化但未取消分配?而且只有当强大且无主的引用计数达到0时,它才会被取消分配吗?
我在一篇有关Medium的文章中阅读了该文章),但我不确定它是否正确。
答案 0 :(得分:4)
首先,让我们知道,这些问题的答案都是我们通常应避免依赖的所有实现细节。现在,进入答案:
ARC是否确实保留了对对象的无主引用计数?
是的,这是真的。每个对象都有三个引用计数:强计数,无所有权计数和弱计数。
始终存储强计数(但以-1的调整存储,因此存储的0表示强参考计数,存储的1表示强参考计数2,依此类推)
还始终存储无主计数,将+1的调整值表示所有强引用,并在取消初始化结束时将其删除。
仅在创建对对象的第一个弱引用后才存储弱引用计数。弱引用计数(如果已存储)将通过+1调整进行存储,该调整表示所有未拥有的引用,并且在对象被释放后将被删除。
那么,如果某个对象的强引用计数达到0,而该对象的未拥有引用计数大于0,则该对象将被取消初始化但未取消分配?
正确。该对象被取消初始化:运行该对象的类的deinit
和所有超类,并且将自身引用的任何对象属性设置为nil。但是,不会释放对象的内存,因为对象的标头必须保持有效,直到对对象的最后一个unowned
引用被破坏为止。
只有当强大且无主的引用计数达到0时,它才会被取消分配吗?
正确。当强引用计数和无所有权引用计数都达到零时,将释放对象。由于大多数对象从未被unowned
引用引用,因此通常是在最后一个强引用被销毁时。
您没有询问弱引用,但是为了完整起见,我也会对其进行解释。当一个对象被(或曾经被)弱引用时,Swift为该对象分配它所谓的“边表条目”(有时也称为“边表”)。
如果对象没有边表,则强计数和无所有权计数将直接存储在对象中,而弱计数(必须为零)不会存储。
如果对象具有边表,则指向该边表的指针将存储在该对象中。强计数,无计数和弱计数以及指向对象的指针都存储在边表中。
对对象的弱引用存储为指向边表而不是对象的指针。这意味着即使仍然存在对对象的弱引用,也可以将其释放(不仅仅是取消初始化)。
如果没有对象的弱引用,则在释放对象时会释放边表。如果仍然存在弱引用,则对象将被释放,但边表仍保持分配状态。当销毁对释放对象的最后一个弱引用时,将取消分配边表。
请注意,当取消初始化或释放Swift对象时,弱引用不会立即设置为nil(已销毁)!仅当程序尝试加载引用或弱引用的容器被取消初始化时,对未初始化对象的弱引用才设置为nil。 (例如,当一个对象具有weak var
属性时,我的意思是“容器”。该对象是weak var
引用的容器。)
RefCount.h
in the Swift source code顶部的大注释解释了所有这些详细信息以及更多信息。
P.S。还有另外一种参考,unowned(unsafe)
,它不会调整任何参考计数。您应尽可能避免使用此类参考(并且几乎总是可以避免的)。
答案 1 :(得分:0)
弱引用和无所有权引用会在一个引用周期中启用一个实例 在不牢牢抓住另一个实例的情况下引用它。
strong
引用将1加上强引用计数。
weak
和unowned
不会不将1增加到强引用计数。
一旦对象具有0个强引用,它将被ARC取消初始化。
如果您尝试访问引用了 取消初始化的对象,您将触发与 强制展开一个nil可选类型。