[Swift]汇总闭包的那些事

用最少的篇幅记录经典概念

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")
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容