Swift3.0P1 语法指南——闭包

原档:https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programming_Language/Closures.html#//apple_ref/doc/uid/TP40014097-CH11-ID94

1、闭包(Closures

闭包是独立的函数代码块,可以在代码中被传递和使用。Swift中的闭包与C语言和Objective-C语言中的block、其他语言中的lambda类似。

闭包可以从上下文中捕获和存储任意变量和常量的引用。这就是所谓的闭合并包裹这些变量和常量。Swift会处理捕获过程中的内存管理。

全局函数和嵌套函数实际上就是闭包的特殊情况。

闭包采取如下三种形式之一:

- 全局函数是一种有名字但不捕获任何值的闭包

- 嵌套函数是一种有名字并且能捕获封闭的函数作用域内的值的闭包。

- 闭包表达式是以轻量级语法写成的、没有名字的、并且能够捕获其上下文的值的闭包。

2、闭包表达式

闭包表达式拥有简洁的风格,在一般场景下可以进行语法优化。

Swift提供了一个sorted(isOrderedBefore:) 方法,其根据用于排序的闭包的返回值为数组的值进行排序。

例如,数组初始化如下:

 let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]

sorted(isOrderedBefore:)方法接收一个排序闭包作为参数,该闭包则接收数组的其中两个元素作为参数并返回一个Bool型的值来决定比较的两个元素哪一个排在前面。

因此,用于排序的闭包的函数类型是:(String, String) -> Bool

则用一般函数的形式写,则是:

 func backward(_ s1: String, _ s2: String) -> Bool {
return s1 > s2
}
var reversed = names.sorted(isOrderedBefore: backward)
// reversed is equal to ["Ewa", "Daniella", "Chris", "Barry", "Alex"]

如果第一个字符串 (s1) 大于第二个字符串 (s2),backward函数返回true,表示在新的数组中s1应该出现在s2前。

对于字符串中的字符来说,“大于” 表示 “按照字母顺序较晚出现”。 这意味着字母"B"大于字母"A",字符串"Tom"大于字符串"Tim"

这个函数将进行字母逆序排序,"Barry"将会排在"Alex"之前。

(1)表达式

用闭包表达式来写这个排序闭包,可以简化成:

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

下面是V2.1的对应方法:

1 reversed = names.sort({ (s1: String, s2: String) -> Bool in
2 return s1 > s2
3 })

闭包的类型为(String, String) -> Bool,闭包的函数体部分由in关键字引出。

由于上面的例子中,函数体部分较短,闭包可以写成:

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

(2)从上下文中推断类型

由于排序闭包是用来作为参数传递给sorted(isOrderedBefore:) 方法的,Swift就能根据上下文推断出闭包的参数类型和返回值类型。

sorted(isOrderedBefore:)方法被String数组调用,则闭包的类型一定是(String, String) -> Bool。则闭包中的参数类型和返回值类型可以忽略不写:

 reversedNames = names.sorted(isOrderedBefore: { s1, s2 in return s1 > s2 } )

实际上任何情况下,通过内联闭包表达式构造的闭包作为参数传递给函数时,都可以推断出闭包的参数和返回值类型,这意味着您几乎不需要利用完整格式构造任何内联闭包。

(3)从单表达式隐式返回

单表达式的闭包可以忽略return关键字,隐式返回单行表达式的结果。

 reversedNames = names.sorted(isOrderedBefore: { s1, s2 in s1 > s2 } )

(4)参数名称缩写

Swift 自动为内联函数提供了参数名称缩写功能,您可以直接通过$0,$1,$2来顺序调用闭包的参数。

如果在闭包表达式中使用参数名称缩写,可以在闭包参数列表中省略对其的定义,并且对应参数名称缩写的类型会通过函数类型进行推断。 in关键字也同样可以被省略:

 reversed = names.sorted(isOrderedBefore: { $ > $ } )

其中,$0和$1分别对应第一个和第二个String参数

(5)运算符函数

Swift 的String类型定义了关于大于号 (>) 的字符串实现,其作为一个函数接受两个String类型的参数并返回Bool类型的值。 而这正好与sorted(isOrderedBefore:)需要的方法的第二个参数需要的函数类型相符合。 因此,可以简单地传递一个>,Swift可以自动推断出您想使用>的字符串函数实现:

 reversed = names.sorted(isOrderedBefore: >)

3、尾随闭包(Trailing Closures)

如果需要将一个很长的闭包表达式作为最后一个参数传递给函数,可以使用尾随闭包来增强函数的可读性。 尾随闭包是一个书写在函数括号之后的闭包表达式,函数支持将其作为最后一个参数调用。

 func someFunctionThatTakesAClosure(closure: () -> Void) {
// function body goes here
} // here's how you call this function without using a trailing closure: someFunctionThatTakesAClosure(closure: {
// closure's body goes here
}) // here's how you call this function with a trailing closure instead: someFunctionThatTakesAClosure() {
// trailing closure's body goes here
}

用尾随闭包,可以将上面的排序闭包写成:

 reversed = names.sorted() { $ > $ }

如果闭包表达式是函数或方法的唯一参数,则可以省略括号:

 reversed = names.sorted { $ > $ }

当闭包非常长以至于不能在一行中进行书写时,尾随闭包变得非常有用。

例如,Swift 的数组有一个map方法,其获取一个闭包表达式作为其唯一参数。 对于数组中的每一个元素,调用一次这个闭包函数,就会返回与该元素所映射的值(可能是不同类型的值)。 具体的映射方式和返回值类型由闭包来指定。

当给数组的每个元素提供闭包函数后,map方法将返回一个新的数组,数组中包含了与原数组一一对应的映射后的值。

下面的例子中,将用map(_:)方法将Int数组映射为String数组。

 let digitNames = [
: "Zero", : "One", : "Two", : "Three", : "Four",
: "Five", : "Six", : "Seven", : "Eight", : "Nine"
]
let numbers = [, , ]

现在,传递一个尾随闭包给numbersmap方法来创建对应的字符串版本数组:

 let strings = numbers.map {
(number) -> String in
var number = number
var output = ""
while number > {
output = digitNames[number % ]! + output
number /=
}
return output
}
// strings is inferred to be of type [String]
// its value is ["OneSix", "FiveEight", "FiveOneZero"]

V2.1:

 1 let strings = numbers.map {
2 (var number) -> String in
3 var output = ""
4 while number > 0 {
5 output = digitNames[number % 10]! + output
6 number /= 10
7 }
8 return output
9 }
10 // strings is inferred to be of type [String]
11 // its value is ["OneSix", "FiveEight", "FiveOneZero"]

map在数组中为每一个元素调用了闭包表达式。 不需要指定闭包的输入参数number的类型,因为可以通过要映射的数组类型进行推断。

闭包表达式在每次被调用的时候创建了一个字符串并返回。 其使用求余运算符 (number % 10) 计算最后一位数字并利用digitNames字典获取所映射的字符串。

注意:在字典中用下标取值时取出的是optional类型。由于这里可以保证number % 10是一个有效的下标,所以用了强制解绑(!)。

4、捕获值

闭包可以在其定义的上下文中捕获常量或变量。 即使定义这些常量和变量的作用域已经不存在,闭包仍然可以在闭包函数体内引用和修改这些值。

在Swift中,最简单的捕获值的闭包形式就是嵌套函数。 嵌套函数可以捕获其外部函数所有的参数以及定义的常量和变量。

下例为一个函数makeIncrementor,它包含了一个嵌套函数incrementor。 嵌套函数incrementor从上下文中捕获了两个值,runningTotalamount。 之后makeIncrementorincrementor作为闭包返回。 每次调用incrementor时,其会以amount作为增量增加runningTotal的值。

 func makeIncrementer(forIncrement amount: Int) -> () -> Int {
var runningTotal =
func incrementer() -> Int {
runningTotal += amount
return runningTotal
}
return incrementer
}

makeIncrementer的返回类型是 () -> Int。

makeIncrementer函数定义了一个整型变量runningTotal(初始为0) 用来存储当前跑步总数。 该值通过incrementor返回。

makeIncrementor有一个Int类型的参数,其外部命名为forIncrement, 内部命名为amount,表示每次incrementor被调用时runningTotal将要增加的量。

incrementor函数用来执行实际的增加操作,单独来看这个函数:

 func incrementer() -> Int {
runningTotal += amount
return runningTotal
}

incrementer函数并没有任何参数,但是在函数体内访问了runningTotalamount变量。这是因为其捕获在包含它的函数体内已经存在的runningTotalamount变量的引用(reference)。捕捉了变量引用,保证了runningTotalamount变量在调用完makeIncrementer后不会消失,并且保证了在下一次执行incrementer函数时,runningTotal可以继续增加。

下面,来使用makeIncrementer:

 let incrementByTen = makeIncrementer(forIncrement: )

定义了一个叫做incrementByTen的常量,该常量指向一个每次调用会加10的incrementor函数。 调用这个函数多次可以得到以下结果:

 incrementByTen()
// returns a value of 10
incrementByTen()
// returns a value of 20
incrementByTen()
// returns a value of 30

创建了另一个incrementor,其会有一个属于自己的独立的runningTotal变量的引用:

 let incrementBySeven = makeIncrementer(forIncrement: )
incrementBySeven()
// returns a value of 7

调用incrementByTen不会影响incrementBySeven中的runningTotal变量:

 incrementByTen()
// returns a value of 40

注意:如果把闭包赋值给一个类实例的属性,并且该闭包通过指向该实例或其成员来捕获该实例,将创建一个在闭包和实例间的强引用环。 Swift 使用捕获列表来打破这种强引用环。

5、闭包是一种引用类型

如果将闭包赋值给了两个不同的常量/变量,两个值都会指向同一个闭包:

 let alsoIncrementByTen = incrementByTen
alsoIncrementByTen()
// returns a value of 50

6、非逃逸闭包(Nonescaping Closures)

一个闭包被当做一个参数传递给一个函数,但这个闭包在函数返回后仍然被调用,这就叫做这个闭包逃离了这个函数

在定义一个需要把闭包作为参数的函数时,可以在参数类型前加上@noescape标记,提示这个闭包不允许逃离函数(不能在函数返回后被调用)。

让闭包带着@noescape标记,可以让编译器在了解闭包的生命周期的情况下做更多的aggressive optimizations。

 func someFunctionWithNonescapingClosure(closure: @noescape () -> Void) {
closure()
}

一种闭包可以逃离函数的方式是,在函数体外将闭包定义为一个变量。例如,很多开始异步操作的函数都接收一个闭包参数作为结束句柄(completion handler),函数在开始这个异步操作后就返回了,但是这个闭包是在异步操作结束之后才被调用,这种情况下,闭包就需要逃离函数(滞后调用)。

 var completionHandlers: [() -> Void] = []
func someFunctionWithEscapingClosure(completionHandler: () -> Void) {
completionHandlers.append(completionHandler)
}

someFunctionWithEscapingClosure(_:)函数接收一个闭包参数,并把这个闭包加到了定义在函数体外的闭包数组中。如果你在这里用了@noescape标记,就会产生编译错误。

将闭包标记为@noescape,可以在闭包中直接引用到self:

 class SomeClass {
var x =
func doSomething() {
someFunctionWithNonescapingClosure { x = }
someFunctionWithEscapingClosure { self.x = }
}
}
let instance = SomeClass()
instance.doSomething()
print(instance.x)
// Prints "200"
completionHandlers.first?()
print(instance.x)
// Prints "100"

7、自动闭包(Autoclosures)

自动闭包是一种自动创建的闭包,其用于绑定一个表达式并将其作为参数传递给函数。自动闭包并不接收任何参数,当它被调用时,它返回绑定在其内部的表达式的值。

通常会调用接收自动闭包的函数,很少去实现这种函数。

例如,assert(condition:message:file:line:)函数接收自动闭包作为condition和message参数。其中,只有在Debug情况下才考虑condition参数,而只有当condition等于false的时候,才执行message。

自动闭包可以允许你惰性求值,内部代码直到调用闭包才会执行。惰性求值可以让你控制何时执行,这在计算消耗很大的情况下非常有用。

 var customersInLine = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
print(customersInLine.count)
// prints "5" let customerProvider = { customersInLine.removeAtIndex() }
print(customersInLine.count)
// prints "5" print("Now serving \(customerProvider())!")
// prints "Now serving Chris!"
print(customersInLine.count)
// prints "4"

直到闭包真正被调用,customersInLine的第一个元素才被移除。需要注意的是:customerProvider的类型不是String,而是()->String。

传递一个闭包作为函数的参数,同样有惰性求值的效果:

 // customersInLine is ["Alex", "Ewa", "Barry", "Daniella"]
func serve(customer customerProvider: () -> String) {
print("Now serving \(customerProvider())!")
}
serve(customer: { customersInLine.remove(at: ) } )
// Prints "Now serving Alex!"

下面是另一版本的serve函数,通过@autoclosure,自动将传入的表达式转换成自动闭包。

 // customersInLine is ["Ewa", "Barry", "Daniella"]
func serve(customer customerProvider: @autoclosure () -> String) {
print("Now serving \(customerProvider())!")
}
serve(customer: customersInLine.remove(at: ))
// Prints "Now serving Ewa!"

@autoclosure属性隐含了@noescape属性,这意味着,这个闭包只能用在函数内部。如果想让闭包能够逃离当前的作用域,在函数返回后闭包还能被调用,则使用 @autoclosure(escaping)属性:

 // customersInLine is ["Barry", "Daniella"]
var customerProviders: [() -> String] = []
func collectCustomerProviders(_ customerProvider: @autoclosure(escaping) () -> String) {
customerProviders.append(customerProvider)
}
collectCustomerProviders(customersInLine.remove(at: ))
collectCustomerProviders(customersInLine.remove(at: ))
print("Collected \(customerProviders.count) closures.")
// Prints "Collected 2 closures."
for customerProvider in customerProviders {
print("Now serving \(customerProvider())!")
}
// Prints "Now serving Barry!"
// Prints "Now serving Daniella!"

在上面的代码中,没有将闭包用作customer参数进行调用,而是将闭包添加到customerProviders数组,这个闭包数组是在函数外部声明的,这意味着,数组中的闭包可以在函数返回后执行。所以,customer参数的值也就可以逃离函数的作用域。

上一篇:常用CSS优化总结——网络性能与语法性能建议


下一篇:JavaScript性能优化