在做惰性变量、内存管理时我们需要弱引用还是无主引用

do we need weak or unowned reference when doing a lazy variable, memory management

我正在通过以下方式在 Swift 中试验内存管理。

class Person {
    let name: String
    //var block: (() -> Void)?

    init(name: String) {
        self.name = name
    }

    //  Declare variable clone, using self inside.
    private lazy var clone: String  = {
        return self.name   
    }()

    deinit {
        print("Destroying \(name)")
    }
    func doSomething() {
        print("Doing something for \(name)")
    }
}

var p2: Person? = Person(name: "p2")
print(p2!.clone)
p2 = nil

如你所见,我在声明惰性变量时在内部使用 self,我认为它仍然可以,因为当 p2 变为 nil 时,我可以看到 deinit 方法被调用.

但是,如果我进行如下更改

    // This is a closure
    private lazy var clone: () -> String  = {
        return self.name    // leaking memory is here
    }

现在,我遇到内存泄漏问题。

我的问题是关于使用惰性实例化的变量,为什么即使我正在使用 self 也没有得到内存泄漏。我想我必须使用它,否则我会得到内存泄漏。

你泄漏内存的原因是因为闭包在默认情况下会强烈捕获 self,同时 self 将闭包作为 属性 强烈持有。跟惰性变量无关。

这里:

 private lazy var clone: () -> String  = {
        return self.name    // leaking memory is here
    }

您正在将闭包本身分配给变量,而不是将它应该分配给 return 的 String。由于您使用的是闭包 保留 self,因此两者永远不会被释放,这会导致 memory leak。当闭包保留为 属性 并且闭包保留自身时,将创建 reference cycle。这是 capture lists 出现的地方。您可以像这样修复泄漏:

private lazy var clone: () -> String  = { [unowned self] in

        return self.name    // leaking memory is fixed
    }

Selfcapture lists 中被声明为 unowned 因为可以安全地假设它 NOTnil 在任何时候。如果您确定变量 永远不会 nil,请使用 unowned,但如果您认为在某些时候它 可能 变成 nil,改用 weak