保留类和结构之间的循环

时间:2017-03-06 17:39:46

标签: swift memory-leaks retain-cycle

假设我有以下代码:

struct X {
    let propertyOfTypeY: Y
}

class Y {
    var propertyOfTypeX: X?
}

let y = Y()
let x = X(propertyOfTypeY: y)
y.propertyOfTypeX = x

如果这些都是类,则意味着保留周期。但是,我不清楚类和结构之间的差异如何应用于上面的示例。它会导致保留周期,还是因为结构的使用而成为安全的代码?

3 个答案:

答案 0 :(得分:7)

是的,你有一个保留周期。

y.propertyOfTypeX = x

将值x复制到y.propertyOfTypeX,包括。{ 属性x.propertyOfTypeY,它是对y的引用。

因此

y.propertyOfTypeX?.propertyOfTypeY === y

成立。你拥有的与

基本相同
class Y {
    var propertyOfTypeY: Y?
}

var y = Y()
y.propertyOfTypeY = y

只有propertyOfTypeYstruct X的一部分 (x还有y}的附加引用。

答案 1 :(得分:4)

TL; DR有一个保留周期,但你可以看到你自己

struct X {
    let propertyOfTypeY: Y
}

class Y {
    var propertyOfTypeX: X?

    deinit {
        print("I was deinit'ed")
    }
}

do {
    let y = Y()
    let x = X(propertyOfTypeY: y)
    y.propertyOfTypeX = x
}
// y and x should be dealloc'ed here, because the "do scope" ends

将会打印出y.propertyOfTypeX = xI was deinit'ed,但如果您执行此任务,则永远不会调用deinit

如果使用闭包,也会发生同样的事情。

答案 2 :(得分:0)

内存图显示参考周期

enter image description here肯定有一个保留周期。

解决方案:应该unownedweak来打破循环

struct X {
    unowned let propertyOfTypeY: Y
}

class Y {
    var propertyOfTypeX: X?

    deinit {
        print("Y deallocated")
    }
}

do {
    let y = Y()
    let x = X(propertyOfTypeY: y)
    y.propertyOfTypeX = x
}