本帖最后由 swifter 于 2016-10-21 04:06 编辑


刚才在Swift大拇指上发了《Swift 3.0 必看:@noescape走了, @escaping来了》,现在继续。

欢迎订阅我在全球最大的Swift开发者社区——敏捷大拇指Swifthumb.com)上的淘帖专辑《Swift 3.0 必看语法变动》,有更多的Swift 3.0语法变动方面的资料,一旦有更新,就会通知到你。

Swift 3.0 必看 系列

Swift 3.0 必看:Swift中被忽略的@noescape - 敏捷大拇指 - Swift 3.0 必看 系列


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

众所周知swift的内存管理是引用计数。闭包里用到的数据都需要捕捉到闭包里,保证闭包执行时这些数据不会被释放还在内存里。Xcode为了让我们意识到闭包里用到的对象其实已经被retain了,就要求我们访问当前属性时显示声明self。

这个时候如果新手就很容易犯引用循环的错误。闭包retain了self,self如果又持有retain了闭包。最后就谁都释放不了,内存就泄露了。

这是Swift中默认闭包的使用场景。

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

这就是非escape闭包。

Swift里针对非escape用@noescape表示。

比如map函数就使用了:

[Swift] 纯文本查看 复制代码
func map<T>(@noescape transform: (Self.Generator.Element) throws -> T) rethrows -> [T]


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




相关链接

Swift 3.0 必看:新增的两种访问控制权限 fileprivate和open

Swift 3.0 必看:集合方法flatten()重命名为joined()

Swift 3.0 必看:typealias支持泛型

Swift 3.0 必看:#selector获取属性的getter和setter方法

Swift 3.0 必看:foundation中数据引用类型改为值类型

Swift 3.0 必看:dynamicType废除,用type(of: )代替

Swift 3.0 必看:给DispatchQueue扩展实现被废Dispatch once原有功能

Swift 3.0 必看:从使用场景了解GCD新API

Swift 3.0 必看:sizeof移进MemoryLayout

Swift 3.0 必看:@noescape走了, @escaping来了

Swift 3.0 必看:Swift中被忽略的@noescape

同时,欢迎订阅我在全球最大的Swift开发者社区——敏捷大拇指Swifthumb.com)上的淘帖专辑《Swift 3.0 必看语法变动》,有更多的Swift 3.0语法变动方面的资料,一旦有更新,就会通知到你。




作者:没故事的卓同学