ARC是否计算了无主参考?

问题描述 投票:3回答:2

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

因此,如果对象的强引用计数达到0并且该对象的无主引用计数大于0,则对象将被取消初始化但不会取消分配?只有当强大且无主的引用计数达到0时才会被取消分配?

我在一篇文章中读到了这篇文章,我认为这是中等的,但我不确定它是否正确。

swift automatic-ref-counting strong-references unowned-references
2个回答
4
投票

首先,让我们知道这些问题的答案都是我们通常应该避免依赖的实现细节。现在,回答:

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

是的,它是真实的。每个对象都有三个引用计数:强计数,无主计数和弱计数。

  • 始终存储强计数(但存储的调整值为-1,因此存储的0表示强引用计数,存储的1表示强引用计数为2,依此类推)。
  • 无主计数也始终存储,调整+1代表所有强引用,并在去初始化结束时删除。
  • 弱引用计数仅在创建对象的第一个弱引用之后存储。弱引用计数(如果已存储)将以+1调整值存储,该调整表示所有无主引用,并在取消分配对象后删除。

因此,如果对象的强引用计数达到0并且该对象的无主引用计数大于0,则对象将被取消初始化但不会取消分配?

正确。取消初始化对象:运行对象类的deinits和所有超类,并且将自身引用的任何对象属性设置为nil。但是,对象的内存未被释放,因为对象的标头必须保持有效,直到对象的最后一个unowned引用被销毁。

只有当强大且无主的引用计数达到0时才会被取消分配?

正确。当强和无主引用计数都达到零时,对象将被释放。由于大多数对象从未被unowned引用引用,因此通常在最后一个强引用被销毁时。

你没有问过弱引用,但为了完整起见,我也会解释它们。当一个对象被弱引用(或曾经被引用过)时,Swift会为该对象分配它所谓的“边表条目”(或者有时只是“边表”)。

  • 如果一个对象没有边表,那么强和无主计数直接存储在对象中,并且不存储弱计数(必须为零)。
  • 如果对象具有边表,则指向边表的指针存储在对象中。强,无主和弱计数以及指向对象的指针存储在边表中。

对象的弱引用存储为指向边表的指针,而不是对象。这意味着即使对它仍然存在弱引用,也可以释放对象(不仅仅是取消初始化)。

如果对象没有弱引用,则在释放对象时释放边表。如果仍有弱引用,则对象将被释放,但是仍然会分配边表。当销毁对释放的对象的最后一个弱引用时,将释放边表。

请注意,当Swift对象被取消初始化或取消分配时,弱引用不会立即设置为nil(已销毁)!当程序尝试加载引用时,或者当弱引用的容器被取消初始化时,对取消初始化对象的弱引用仅设置为nil。 (我所说的“容器”是指,例如,当一个对象具有weak var属性时。该对象是weak var引用的容器。)


RefCount.h in the Swift source code顶部的一条大评论解释了所有这些细节等等。


附:还有一种参考,unowned(unsafe),它不会调整任何引用计数。如果可能的话,你应该避免这种引用(并且几乎总是可以避免)。


0
投票

弱引用和无引用引用使引用周期中的一个实例能够引用另一个实例而不保持强大的保持。

strong引用将强引用计数加1。

weakunowned不会在强引用计数中加1。

只要一个对象有0个强引用,它就会被ARC取消初始化。

如果您尝试访问引用已取消初始化对象的无主属性,则会触发与强制解包nil可选类型相当的运行时错误。

文件:ARC Documentation

ARC and Memory Management in Swift

© www.soinside.com 2019 - 2024. All rights reserved.