对我们这种第一门语言是C语言的人来说,Closure难免很陌生,且比较难掌握。另外,现在网络上很多资料都是根据苹果官方文件翻译过来的,基本相同。
据说,Closure与Objective-C中的Block有相似之处。不幸的是,我没有使用过Objective-C。希望使用过Objective-C的朋友一起来讨论,哈哈哈。
接下来,我就把我自己在学习中里理解写出来,希望对大家有所帮助
引入
过去,我们对函数很熟悉(如果Swift是你所学习的第一门编程语言也无妨,参见Swift Programming Language 的 Function 部分。现在,我假设你已经了解了函数)。在下面的例子中,我实现了排序功能,这样是苹果官方教程中的例子,现在我们引用它。
首先,我们定义一个String数组
let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
我们使用sort函数来对names数组进行排序。backwards韩素好根据字母顺序对Strings类型数据排序,例如, “B” > “A”, “Tom” > “Tim”。
func backwards(s1: String, s2: String) -> Bool {
return s1 > s2
}
var reversed = sorted(names, backwards)
// reversed 为 ["Ewa", "Daniella", "Chris", "Barry", “Alex"]
显然,backwards函数的功能为判断两个参数s1, s2的大小关系。当s1 > s2时,函数返回true;否则,函数返回false
这样简单的功能可能不会让你对上面例子中的写法反感。实际上,这是一个相当冗长的写法,当你实现一个更复杂的功能时,闭包就是你更好的选择。
闭包的语法 Closure Expression Syntax
现在我们介绍闭包的语法,这会使你对闭包有初步的认识,也会帮助你更容易了解句法之后的知识
{ (parameters) -> returnType in
statements
}
闭包表达式语法可以使用常量、变量和inout类型作为参数,不提供默认值。 也可以在参数列表的最后使用可变参数。 元组也可以作为参数和返回值。
下面的例子展示了把之前的backwards功能写成闭包形式
reversed = sorted(names, { (s1: String, s2: String) -> Boolin return s1 > s2 })
值得注意的是, (s1: String, s2: String) -> Bool 这一部分与函数的声明部分相同;return s1 > s2 这部分是该闭包的功能,即判断两个参数s1, s2的大小关系(同上面的backwards函数相同);闭包的函数体部分由关键词引入;区别于函数定义语法的是,闭包中参数、返回类型都写在{}中。
根据上下文推断类型 Inferring type from context
Swift一个很强大的特点是type inference。在闭包中,Swift可以根据传入的参数判断参数类型,因此,你在构造闭包的时候,诸如String, Int等不是闭包表达定义式中的必要部分;同样,返回类型也不是必要部分。在下面的例子中,我们修改了在上一片段中的闭包表达式
reversed = sorted(names, { s1, s2 in return s1 > s2 })
单表达式闭包隐式返回 Implicit return single-expression closures
一些时候,Swift可以判定关键词in后面的部分为返回部分。也就是说,return也可以被省略掉。这种省略关键词return的方法叫做单表达式闭包隐式返回。上面的例子可写成如下形式:
reversed = sorted(names, { s1, s2 in s1 > s2 } )
在这个例子中,sorted函数的第二个参数类型明显表明了其中闭包表达式必须返回一个Bool值(Boolean);且闭包表达式中s1 > s2显然返回Bool类型值。因此这里没有歧义,return可以省略。
参数名称缩写 Shorthand argument names
Swift提供了参数名称缩写功能,你可以使用$0, $1, $2 等调用闭包参数,$0, $1, $2 等顺序指代闭包中表示的参数
如果你使用参数名称缩写,参数的定义也可以被缩写,同是,参数的类型也可以被判断。关键词in同样可以被省略,因为此时闭包表达式完全由闭包函数体构成:
reversed = sorted(names, { $0 > $1 } )
在这个例子中,$0, $1表示闭包中的第一个和第二个String参数
运算符函数 Operator functions
Swift是一门惊人的语言,我们可以写出来简短简短更简短的代码(我是果粉,原谅我对Swift的赞扬,
)。Swift存在接受两个String值的>函数,且返回值类型为Bool(func ><T : _Comparable>(lhs: T, rhs: T) -> Bool),这正好与sorted中第二个参数的类型相同。因此,你可以只将>作为参数,Swift可以自动判断出>的函数实现
reversed = sort(names, >)
额外的例子 Extra example
如果你对闭包仍有不解之处,下面的例子或许可以帮到你。
下面的例子的目的是将数组中的数字求和
//待更新