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