金氧

Swift中被忽略的@noescape

这里需要先介绍一下escape的概念。当一个闭包当做一个参数传进函数里,这个闭包是在这个函数执行完后执行的,这个时候我们就说这个闭包从函数逃出来了(escape)。这种场景很常见,比如我们进行一个异步的请求,请求时会传入一个handler,比如当请求成功后执行达到回调的目的。

众所周知swift的内存管理是引用计数。闭包里用到的数据都需要捕捉到闭包里,保证闭包执行时这些数据不会被释放还在内存里。Xcode为了让我们意识到闭包里用到的对象其实已经被retain了,就要求我们访问当前属性时显示声明self。
这个时候如果新手就很容易犯引用循环的错误。闭包retain了self,self如果又持有retain了闭包。最后就谁都释放不了,内存就泄露了。

这是swift中默认闭包的使用场景。
但是这里是有另外一种可能,假设有一个闭包是传入用于sort用的,或者比如作为map参数的闭包。当这行代码执行完成时,这个闭包也就使用完了,之后不会再被执行。这个情况下,闭包就不必再持有里面用到的对象。
这就是非escape闭包。

swift里针对非escape用@noescape表示。
比如map函数就使用了:

func map<T>(@noescape transform: (Self.Generator.Element) throws -> T) rethrows -> [T]

这样标记之后能看到的好处就是这个闭包里如果再使用self的属性不需要加self.了。对于编译器而言,在知道是noescape闭包后可以进行一些内存的优化。