7

谁能解释一下为什么这不会泄漏?

self在 a 内捕获,closure因此我将有两个指向对方的强指针,因此,deinit永远不应为 Person 对象调用该消息。

首先,这是我的班级Person

class Person {
    var name: String
    init(name: String) { self.name = name }
    deinit { print("\(name) is being deinitialized") }
}

这是我的 ViewController 的实现

class ViewController: UIViewController {

    var john:Person?

    func callClosureFunction( closure:(name:Bool) -> () ) {
        closure(name: true)
    }

    override func viewDidLoad() {
        super.viewDidLoad()

        john = Person(name:"John")

        self.callClosureFunction { (name) in

            self.john?.name = "John Appleseed"
            self.john = nil

            // xcode prints - John Appleseed is being deinitialized
        }

    }

}

我希望能够通过以下方式解决问题:

self.callClosureFunction { [weak self] (name) in ...

但这甚至没有必要。为什么?

4

3 回答 3

6

由于您的视图控制器没有保留闭包,因此没有循环引用。如果你这样写:

class ViewController: UIViewController {

    var john:Person?
    var closure:(Bool)->()? 

    func callClosureFunction( closure:((name:Bool) -> ())? ) {
        closure?(name: true)
    }

    override func viewDidLoad() {
        super.viewDidLoad()

        john = Person(name:"John")
        closure = { (name) in

            self.john?.name = "John Appleseed"    

            // Because this closure will never be released, the instance of Person will never deinit either
        }
        self.callClosureFunction(closure) 
    }  
}

然后视图控制器将保留闭包,而闭包将通过其对self. 因此,两者都不会被释放,并且如果您没有显式设置self.john = nil(您在原始示例中所做的),那么该Person实例将永远不会被deninit调用。

在不必要的情况下不恰当地在闭包中使用弱是很常见的self(这实际上会导致一些晦涩的错误)。要记住的关键规则是弱引用通常不是 ARC 下的默认值。Strong 应该是默认值,除非它会导致一个保留循环,在这种情况下,弱应该只用于破坏该循环引用。闭包也一样:strongself应该是默认值,除非self这种情况下,它也对闭包本身有强引用。

于 2016-08-10T16:12:22.773 回答
0

您正在捕获self指向 的哪些点ViewController,但您想知道该Person实例。

Person实际上不是循环引用的,因此当您在关闭结束时将其设置为 nil 时,它会被取消初始化并释放。

实施并看看它是如何工作的deinitViewController

于 2016-08-10T15:33:09.847 回答
0

我在闭包中捕获 self ,因此我将有两个指向彼此的强指针,因此,永远不应为 Person 对象调用 deinit 消息。

不,你有一个强指针,从闭包到self. 没有从闭包返回到self. 因此,您有一个有向无环图,这对 ARC 来说没有问题。

但是,您的实验从一开始就存在缺陷。即使闭包被捕获,John Appleseed Person对象仍然会deinit. 该对象的生命周期完全依赖于john来自您的ViewController. 当您将该引用设置为 时nil,您将删除对该John Appleseed对象的最后一个引用,因此它被取消初始化。

于 2016-08-10T16:20:43.910 回答