考虑这个类:
class A { var closure: (() -> Void)? func someMethod(closure: () -> Void) { self.closure = closure } }
someMethod将传入的闭包分配给类中的属性。
现在来了另一个类:
class B { var number = 0 var a: A = A() func anotherMethod() { a.someMethod { self.number = 10 } } }
如果我调用另一个方法,那么闭包{self.number = 10}将被存储在A的实例中。由于在闭包中被捕获,所以A的实例也将被强力引用。
这基本上是一个逃脱关闭的例子!
你可能想知道,“什么?那么关闭从哪里逃出来了?
闭包从方法的范围转移到类的范围。而且可以稍后再来,甚至在另一个线程上!如果处理不当,可能会导致问题。
为了避免意外逃脱关闭并导致保留周期和其他问题,请使用@noescape属性:
class A { var closure: (() -> Void)? func someMethod(@noescape closure: () -> Void) { } }
现在如果你试图写self.closure =关闭,它不会编译!
更新:
在Swift 3中,默认情况下,所有闭包参数都不能退出。您必须添加@escaping属性才能使闭包能够从当前作用域中转义。这为您的代码增加了更多的安全性!