更新至Swift 3.0
基本語法
Closures 在 Swift 中的概念類似 C 和 Objective-C 中的 blocks 和其它語言中的 lambdas.
{ (參數) -> return type in
表達式
}
舉例:排序以下字符串數組
let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
普通方法調用
func backwards(s1: String, _ s2: String) -> Bool {
return s1 > s2
}
var reversed = names.sort(backwards) // 系統提供的sort方法
// reversed 等于 ["Ewa", "Daniella", "Chris", "Barry", "Alex"]
將自定義 backwards 方法作為參數傳入 sort(_:) 方法用于排序
閉包調用
reversed = names.sort({ (s1: String, s2: String) -> Bool in
return s1 > s2
})
- 閉包寫法和普通方法的區別
- 可以去掉 func 聲明 和方法名, 將之后內容用 { } 包起來
- 由于參數后的 { 通過上步驟挪到前面去了, in 這個關鍵字可表示閉包內容的開始
- 其它照舊,這么短的方法可寫成一行,簡潔
根據 names 的 string 內容,和 sort 方法聲明需要的參數
系統可自動推導出這個閉包的參數類型,所以可不寫參數類型
reversed = names.sort( { s1, s2 in return s1 > s2 } )
閉包會把單行表達式的結果隱式 return 所以 return 也可以不用寫
reversed = names.sort( { s1, s2 in s1 > s2 } )
Swift 提供了參數名的縮寫,按順序分配分別為 $0,$1,$2 以此類推
reversed = names.sort( { $0 > $1 } )
Swift 做了一些工作讓代碼可以使用 Operator Functions 讓代碼變的更簡單
reversed = names.sort( { > } )
Trailing Closures | 尾隨閉包
如果閉包作為參數傳遞給方法時,在最后一位,它允許被寫在 ( ) 外面
reversed = names.sort() { $0 > $1 }
要是這方法只有這么一個閉包參數,那連括號都可以省掉了
reversed = names.sort { $0 > $1 }
其存在的意義是讓代碼更簡潔,可讀性更好
Capturing Values | 捕獲數值
一個閉包可以獲取在它周圍上下文定義的常量或變量,哪怕這參數不存在了也依然可以
閉包屬于引用類型,把閉包返回給一個對象時,如果閉包內有通過引用該對象,去方法其屬性變量之類的操作,就會產生相互引用。從而導致的循環引用,會使內存無法釋放
Nonescaping Closures | 非逃逸閉包
當一個閉包被當做參數傳遞給一個方法時,而且它會在這個方法返回后被調用,那它就算是個逃逸閉包
對于非逃逸的閉包,可以在閉包參數前聲明關鍵字 @noescape 來告訴編譯器,于是編譯器就會做更多更激進的優化
func someFunctionWithNonescapingClosure(closure: @noescape () -> Void) {
closure()
}
Autoclosures | 自動閉包
當我們把一個閉包作為參數傳遞的時候,這個閉包有一對醒目的花括號來表示它是個閉包。哪怕它只有一行表達式
// customersInLine is ["Alex", "Ewa", "Barry", "Daniella"]
func serveCustomer(customerProvider: () -> String) {
print("Now serving \(customerProvider())!")
}
serveCustomer( { customersInLine.removeAtIndex(0) } )
// Prints "Now serving Alex!”
自動動閉包會把傳遞給函數的表達式包自動裝成閉包,這樣就不用每次都寫一對 {} 了,只要在參數前聲明 @autoclosure 關鍵字即可,但是這個閉包不接受任何參數,只計算閉包內的內容
// customersInLine is ["Ewa", "Barry", "Daniella"]
func serveCustomer(@autoclosure customerProvider: () -> String) {
print("Now serving \(customerProvider())!")
}
serveCustomer(customersInLine.removeAtIndex(0))
// Prints "Now serving Ewa!”
自動閉包隱式的就是個非逃逸閉包,如果要傳遞的是逃逸閉包,需要加上聲明 @autoclosure(escaping) 即可
參考