用最少的篇幅记录经典概念
1.闭包表达式语法
{
(参数表) -> 返回值类型 in
...
}
2.简化
let names = ["Chris","Alex","Ewa"]
基本写法
var result = names.sorted(by: {s1:String,s2:String) -> Bool in
return s1 > s2
})
省去参数和返回值声明
var result = names.sorted(by: {s1,s2 in return s1 > s2 })
再省去return
var result = names.sorted(by: {s1,s2 in s1 > s2 })
再将参数名缩写并省去
var result = names.sorted(by: {$0 > $1 })
运算符方法>正好满足sorted的函数参数类型,严格说这不算省略,>本身就是一个函数名。看起来简洁而已
var result = names.sorted(by: > )
尾随闭包
闭包表达式做最后一个参数时,可以把闭包放在函数参数括号之后。也不用写参数标签。
var result = names.sorted() {$0 > $1 } //去掉了 by:
如果是唯一参数,还可以省掉()
var result = names.sorted {$0 > $1 }
3.值捕获
这个在其他语言里不是什么新鲜事:
闭包里可以访问和修改上下文(闭包之外定义)的变量、常量,哪怕他们的作用域已经不存在了。
但是,一旦要解决闭包引起的循环强引用时,就要再次提到值捕获。比如闭包非常典型的会和self实例产生相互引用导致无法自动销毁。可以把这些引用放到'捕获列表'中,声明为非强引用(是弱的或者无主的)
//代码片段
class X {
var asHTML : Void -> String = {
[unowned self , weak otherProp = self.otherProp ] in //捕获列表降低强引用
if let text = self.text {
return ... //伪代码
} else ... //伪代码
}
}
4.逃逸闭包
闭包作为参数传入函数但是在函数返回之后才执行就是逃逸闭包。函数的参数要加@escaping. 逃逸闭包中要显式引用self.
带?的不能加@escaping
class Student{
private var doWork:(()->())?
//init(delegate: (()->())?){
init(delegate: @escaping ()->()){
self.doWork = delegate
}
deinit {
print("deinit student")
}
}