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