- 阅读Swift中文文档所做笔记
闭包的简单概念
- 闭包是可以在你的代码中被传递和引用的功能性独立模块,和Python中的匿名函数类似。
闭包能够捕获和存储定义在其上下文中的任何常量和变量的引用,这也就是所谓的闭合并包裹那些常量和变量,因此被称为“闭包”,Swift 能够为你处理所有关于捕获的内存管理的操作。
- 闭包符合以下三种形式中的一种
全局函数是一个有名字但不会捕获任何值的闭包;
内嵌函数是一个有名字且能从其上层函数捕获值的闭包;
闭包表达式是一个轻量级语法所写的可以捕获其上下文中常量或变量值的没有名字的闭包。
- 本篇主要介绍闭包表达式及其优化改良
闭包表达式
闭包表达式是为了使代码简短清晰所做的一系列语法优化,下边例子通过使用几次迭代展示sorted(by:)
方法的精简来展示这些优化,每一次都让相同的功能性更加简明扼要。
- 一般形式:
{ (parameters) -> (return type) in
statements
}
-
sorted(by:)
是Swift标准库提供的排序函数,这里不做赘述。
下面这个闭包表达式的栗子使用 sorted(by:) 方法按字母排序顺序来排序一个 String 类型的数组。这是将被排序的初始数组:
let names = ["Bryant", "LeBorn", "Chris", "Harden", "Wade"]
-
sorted(by:)
接收一个闭包,该闭包包含两个和传入的数组中内容类型相同的参数,返回一个Bool
类型的值,用来表示排序后第一个值应该出现在第二个值得前还是后,前则返回True
, 后则返回False
。
let names = ["Bryant", "LeBorn", "Chris", "Harden", "Wade"]
func backwords (_ s1: String, _ s2: String) -> Bool{
return s1>s2
}
let newname = names.sorted(by: backwords)
print(newname)
排序后内容:
["Wade", "LeBorn", "Harden", "Chris", "Bryant"]
- 用闭包来改写:
let newname2 = names.sorted(by: {(s1: String, s2: String) -> Bool in
return s1<s2 //注意此处是 <
})
print(newname2)
排序后内容
["Bryant", "Chris", "Harden", "LeBorn", "Wade"]
- 改良-1
- 由于
sorted(by:)
接收的闭包的参数是已知类型的数组,因此它的参数可以从上下文中自动判断出来。 - 而闭包函数体如果较为简短的话也可以写在一行,所以闭包可以改写成:
let newname2 = names.sorted(by: {(s1, s2) -> Bool in return s1<s2})
print(newname2)
- 改良-2 单表达式闭包隐式返回
sorted(by:)
接收闭包的结构包涵返回 Bool 值的单一表达式 (s1 > s2),因此 return 关键字能够被省略:let newname3 = names.sorted(by: {(s1, s2) in s1<s2})
- 改良-3 Swift 自动对行内闭包提供简写实际参数名
你可以用$0,$1,$2,表示第一二三个参数:
let newname3 = names.sorted(by: {$0 < $1})
- 改良-4 运算符函数
Swift的String
类型定义了关于>,<符号的特定实现,让其作为有两个String
类型的参数并且返回一个Bool
类型的值的函数,这与sorted(by:)
所要接受的闭包恰好符合,因此我们可以把代码改成:
let final = names.sorted(by: < )
小结
- 闭包常见优化:
- 利用上下文推断形式参数和返回值的类型
- 单表达式的闭包可以隐式返回
- 简写实际参数名