尝试理解Swift中的@escaping

要理解@escaping,首先需要理解closure, 要理解closure,首先理解匿名函数。

先理解匿名函数

要在Swift中构造匿名函数,需要:

  1. 创建函数体,包括花括号,但是不需要函数声明
  2. 如果必要,将函数的参数列表与返回类型作为花括号中的第一行,后跟关键字in.

例子1: 将以下函数变成匿名函数:

func whatToAnimate(){
    self.myButton.frame.origin.y += 20
}

匿名函数版本:

{
    () -> () in
    self.myButton.frame.origin.y += 20
}

例子2: 有参数的具名函数:

func whatToDoLater(finished: Bool){
    print("finished: \(finished)")
}

匿名函数版本:

{
    (finished: Bool) -> () in
    print("finished: \(finished)")
}

可省略的地方:

  1. 省略返回类型
  2. 没有参数可以省略 in 这一样
  3. 省略参数类型
  4. 省略圆括号,这个是如果就一个参数,并且我们编译器可以推断出其类型的话
  5. 省略 in, 直接用名字比如 $0
  6. 省略参数名,用_代替
  7. 省略函数实参标签(又叫尾函数
    ...

一路省略下来,匿名函数真是节约。

再理解闭包

理解匿名函数之后,我们来看闭包的表达式:

{ (parameters) -> return type in
    statements
}

这就是匿名函数啊,而我们叫它闭包则是因为:

闭包可以捕获和存储其所在上下文中任意常量和变量的引用。 这就是所谓的闭合并包裹着这些常量和变量,俗称闭包。

而在这一篇中,我们可以一步一步看到如何从

reversed = sorted(names, { (s1: String, s2: String) -> Bool in
    return s1 > s2
})

利用省略到:

reversed = sorted(names, >)

再利用 trailling closure的特性,可以这样:

reversed = sorted(names) { $0 > $1 }

闭包当然还有更好的例子,因为我们来看它的一些特性。

Escaping Closure(逃逸闭包

如果一个闭包被作为一个参数传递给一个函数,并且在函数return之后才被唤起执行,那么这个闭包是逃逸闭包。

好抽象的描述,这篇文章写得不错:

不逃逸闭包的生命周期:

  1. Pass a closure into a function
  2. The function runs the closure (or not)
  3. The function returns

然后这个closure就死掉了。

而逃逸闭包则会一直存在,没有因为function被kill而死掉,所以我们叫这个闭包为逃逸闭包(?

在以下的两种情况,我们需要使用 escaping closure:

  1. 异步execution,并不能说函数return了就把closure kill掉,因为这个closure可能还没有执行完毕
  2. 存储,如果任何全局变量都有一些些存储存在,那么这个closure也被逃逸掉了。

在 Swift 1 和 2中, closure by default 是 escaping的,所以我们需要用 @noescape 来mark.
在 Swift 3中,closure by default是non-escaping,我们需要用@escaping 来mark

In Swift 3, closure parameters are non-escaping by default; you can use the new @escaping attribute if this isn’t what you want. Non-escaping closures passed in as arguments are guaranteed to not stick around once the function returns.

而一般情况下,弱🐔就跟着Xcode的提示走?

参考:

  1. <iOS 9 Programming Fundamentals with Swift> by Matt Neuburg
  2. 闭包
  3. Escaping and Nonescaping Closures in Swift 3

推荐阅读更多精彩内容

  • 86.复合 Cases 共享相同代码块的多个switch 分支 分支可以合并, 写在分支后用逗号分开。如果任何模式...
    无沣阅读 1,157评论 1 5
  • 以下翻译自Apple官方文档,结合自己的理解记录下来。翻译基于 swift 3.0.1 原文地址 Closure...
    艺术农阅读 1,147评论 0 3
  • 前言:swift是一门很优雅,很简洁,功能很强大的语言,同时也是一门很复杂的语言。进门比较简单,看完苹果官方的文档...
    一剑孤城阅读 5,407评论 0 10
  • 爱情的炙热,如同滑过天空的流星,那样绚丽迷人。伴着融化一切的热,在天空中展现出那世间最美的景色 流星的热可以譬美永...
    雨夜聆听者阅读 135评论 0 0
  • 文/叶丹颖 1 你是原点 从你的坐标点出发 我开始重新打探这世界 你是我爱上这个世界的全部理由 2 我是个只愿意对...
    树姑娘阅读 339评论 0 4