原文网址:http://www.yiibai.com/swift/closure_expressions.html
闭包表达式(Closure Expressions)
嵌套函数 是一个在较复杂函数中方便进行命名和定义自包含代码模块的方式。当然,有时候撰写小巧的没有完整定义和命名的类函数结构也是很有用处的,尤其是在您处理一些函数并需要将另外一些函数作为该函数的参数时。
闭包表达式是一种利用简洁语法构建内联闭包的方式。 闭包表达式提供了一些语法优化,使得撰写闭包变得简单明了。 下面闭包表达式的例子通过使用几次迭代展示了sort
函数定义和语法优化的方式。 每一次迭代都用更简洁的方式描述了相同的功能。
sort
函数,会根据您提供的基于输出类型排序的闭包函数将已知类型数组中的值进行排序。 一旦排序完成,函数会返回一个与原数组大小相同的新数组,该数组中包含已经正确排序的同类型元素。
下面的闭包表达式示例使用sort
函数对一个String
类型的数组进行字母逆序排序,以下是初始数组值:
let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
sort
函数需要传入两个参数:
sort
函数当排序结束后传入的第一个参数排在第二个参数前面还是后面。如果第一个参数值出现在第二个参数值前面,排序闭包函数需要返回true
,反之返回false
。
该例子对一个String
类型的数组进行排序,因此排序闭包函数类型需为(String, String) -> Bool
。
提供排序闭包函数的一种方式是撰写一个符合其类型要求的普通函数,并将其作为sort
函数的第二个参数传入:
func backwards(s1: String, s2: String) -> Bool {
return s1 > s2
}
var reversed = sort(names, backwards)
// reversed 为 ["Ewa", "Daniella", "Chris", "Barry", "Alex"]
如果第一个字符串 (s1
) 大于第二个字符串 (s2
),backwards
函数返回true
,表示在新的数组中s1
应该出现在s2
前。 对于字符串中的字符来说,“大于” 表示 “按照字母顺序较晚出现”。 这意味着字母"B"
大于字母"A"
,字符串"Tom"
大于字符串"Tim"
。 其将进行字母逆序排序,"Barry"
将会排在"Alex"
之后。
然而,这是一个相当冗长的方式,本质上只是写了一个单表达式函数 (a > b)。 在下面的例子中,利用闭合表达式语法可以更好的构造一个内联排序闭包。
闭包表达式语法有如下一般形式:
{ (parameters) -> returnType in
statements
}
闭包表达式语法可以使用常量、变量和inout
类型作为参数,不提供默认值。 也可以在参数列表的最后使用可变参数。 元组也可以作为参数和返回值。
下面的例子展示了之前backwards
函数对应的闭包表达式版本的代码:
reversed = sort(names, { (s1: String, s2: String) -> Bool in
return s1 > s2
})
需要注意的是内联闭包参数和返回值类型声明与backwards
函数类型声明相同。 在这两种方式中,都写成了(s1: String, s2: String) -> Bool
。 然而在内联闭包表达式中,函数和返回值类型都写在大括号内,而不是大括号外。
闭包的函数体部分由关键字in
引入。 该关键字表示闭包的参数和返回值类型定义已经完成,闭包函数体即将开始。
因为这个闭包的函数体部分如此短以至于可以将其改写成一行代码:
reversed = sort(names, { (s1: String, s2: String) -> Bool in return s1 > s2 } )
这说明sort
函数的整体调用保持不变,一对圆括号仍然包裹住了函数中整个参数集合。而其中一个参数现在变成了内联闭包(相比于backwards
版本的代码)。
因为排序闭包函数是作为sort
函数的参数进行传入的,Swift可以推断其参数和返回值的类型。 sort
期望第二个参数是类型为(String, String) -> Bool
的函数,因此实际上String
,String
和Bool
类型并不需要作为闭包表达式定义中的一部分。 因为所有的类型都可以被正确推断,返回箭头 (->
) 和围绕在参数周围的括号也可以被省略:
reversed = sort(names, { s1, s2 in return s1 > s2 } )
实际上任何情况下,通过内联闭包表达式构造的闭包作为参数传递给函数时,都可以推断出闭包的参数和返回值类型,这意味着您几乎不需要利用完整格式构造任何内联闭包。
单行表达式闭包可以通过隐藏return
关键字来隐式返回单行表达式的结果,如上版本的例子可以改写为:
reversed = sort(names, { s1, s2 in s1 > s2 } )
在这个例子中,sort
函数的第二个参数函数类型明确了闭包必须返回一个Bool
类型值。 因为闭包函数体只包含了一个单一表达式 (s1 > s2
),该表达式返回Bool
类型值,因此这里没有歧义,return
关键字可以省略。
Swift 自动为内联函数提供了参数名称缩写功能,您可以直接通过$0
,$1
,$2
来顺序调用闭包的参数。
如果您在闭包表达式中使用参数名称缩写,您可以在闭包参数列表中省略对其的定义,并且对应参数名称缩写的类型会通过函数类型进行推断。 in
关键字也同样可以被省略,因为此时闭包表达式完全由闭包函数体构成:
reversed = sort(names, { $0 > $1 } )
在这个例子中,$0
和$1
表示闭包中第一个和第二个String
类型的参数。
实际上还有一种更简短的方式来撰写上面例子中的闭包表达式。 Swift 的String
类型定义了关于大于号 (>
) 的字符串实现,其作为一个函数接受两个String
类型的参数并返回Bool
类型的值。 而这正好与sort
函数的第二个参数需要的函数类型相符合。 因此,您可以简单地传递一个大于号,Swift可以自动推断出您想使用大于号的字符串函数实现:
reversed = sort(names, >)