Swift异步执行的关键在于Closures,其具体语法如下:###
{ (parameters) -> return type in
statements
}
首先,拿简单的排序来简单说明一下:(无聊请略过)###
- 字符串排序,定义排序方法
let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"] //将要排序的数组
func backward(_ s1: String , _ s2: String ) -> Bool{
return s1 > s2 //倒序
}
2.借助Swift自带排序函数,其语法如下:
names.sorted(by: (String, String) -> Bool) //接收两个string参数返回Bool值,与上面定义的函数对应
3.那么排序可这么执行:
var reversedNames = names.sorted(by: backward)
输出结果:["Ewa", "Daniella", "Chris", "Barry", "Alex"]
4.使用Closures套用其语法可以这样写:
reversedNames = names.sorted(by: {(_ s1: String, _ s2: String) -> Bool in
return s1 > s2
})
5.因为根据names的类型可以推断出参数的类型和返回值,并且如果Closures里只有一条语句可以省略return,可以简写为:
reversedNames = names.sorted(by: { s1,s2 in s1 > s2})
6.由于Swift在带有快速参数,分别是$0,$1...分别代表第一个参数、第二个参数等等,因此我们可以进一步简化为:
reversedNames = names.sorted(by: { $0 > $1})
7.另外,为了方便,Swift专门对字符串的比较定义了方法(>或<),其结构就是接受两个字符串,返回true或者是false,所以我们也可以这样去写:
reversedNames = names.sorted(by: >) //倒序
reversedNames = names.sorted(by: <) //升序
异步说明###
假设我们在做一个下载器,创建了一堆任务进行下载,那么在任务完成的时候我们要对用户进行提醒,比如吐司或者是弹框亦或是本地通知提醒。这是很自然的,我们会用到Closures。于是我们这样去写:
func startDownloadTask(completionHandler: () -> Void){
completionHandler() //当startDownloadTask return之后执行completionHandler
}
这样去写单个的下载不会有太大的问题。试想一下,如果要求你在所有的下载结束之后汇总进行提示呢?那我们就不能这样去写了,起码单个下载完成之后不应该立刻就执行completionHandler,这是就要跳出Closures,Swift上称为Escaping Closures。改造上面的方法如下:
var completionHandlers: [() -> Void] = []
func someFunctionWithEscapingClosure(completionHandler: @escaping () -> Void ) {
completionHandlers.append(completionHandler)
}
但下载队列中没有任务在执行了之后,在处理所有的completionHandlers。
附上Swift3.0的改进:
DispatchQueue.global().async {}
DispatchQueue.main.async(execute: {refreshView.endLoading()})
}