ARC是否保存计数以备不时之需?

时间:2019-02-22 23:47:34

标签: swift automatic-ref-counting strong-references unowned-references

ARC是否确实保留了对对象的无主引用计数?

那么,如果某个对象的强引用计数达到0而该对象的未拥有引用计数大于0,则该对象将被取消初始化但未取消分配?而且只有当强大且无主的引用计数达到0时,它才会被取消分配吗?

我在一篇有关Medium的文章中阅读了该文章),但我不确定它是否正确。

2 个答案:

答案 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加上强引用计数。

weakunowned不会将1增加到强引用计数。

一旦对象具有0个强引用,它将被ARC取消初始化。

  

如果您尝试访问引用了   取消初始化的对象,您将触发与   强制展开一个nil可选类型。

文档: ARC Documentation

ARC and Memory Management in Swift